/
My Favorite Patients MVC Example

My Favorite Patients MVC Example

The Goal

We're going to add a "My Favorite Patients" feature in our module. We'll add a link in the OpenMRS header that says "My Favorite Patients" and that page will show a list of the currently logged-in user's favorite patients. We will store these using the existing "Cohort" object, rather than creating our own domain object:

Initial Setup

We're going to do this within the jspexample module. If we were doing this for real, we'd create a new module, and add the webModuleApplicationContext.xml file as described here

Create a page to show My Favorite Patients

Note that we're going to delegate the actual work to a MyFavoritePatientUtil class. In real life we would make a service for this.

@Controller class MyFavoritePatientsController { @RequestMapping("/module/jspexample/myfavoritepatients") public void showFavoritePatients(ModelMap model) { model.put("favorites", MyFavoritePatientUtil.getFavoritePatients(Context.getAuthenticatedUser())); model.put("me", Context.getAuthenticatedUser()); } }

Next we need a JSP to display this, in the standard location of /web/module/myfavoritepatients.jsp:

<%@ include file="/WEB-INF/template/include.jsp" %> <%@ include file="/WEB-INF/template/header.jsp" %> <h1>My ${fn:length(favorites)} favorite patient(s) (for ${me.username})</h1> <ul> <c:forEach var="p" items="${favorites}"> <li> ${p.personName} <c:choose> <c:when test="${p.gender == 'F'}"> Female </c:when> <c:otherwise> Male </c:otherwise> </c:choose> </li> </c:forEach> </ul> <%@ include file="/WEB-INF/template/footer.jsp" %>

The only problem is that this list is empty. Let's make a hacky form that allows us to add a patient

A hacky way to add new favorite patients

First we add this to the JSP page

<form method="post" action="addFavoritePatient.form"> Add a patient by primary key: <input type="text" name="patient_id"/> <input type="submit" value="Add"/> </form>

Better widget for picking a Patient

<form method="post" action="addFavoritePatient.form"> Add a patient <openmrs:fieldGen formFieldName="patient_id" type="org.openmrs.Patient"/> <input type="submit" value="Add"/> </form>

Now we need a method on the controller to handle that submission.

  • You can use the @RequestParam annotation to have Spring automatically bind a parameter in the request to an argument to your method. It automatically converts from String (which all HTTP request parameters are by nature) to the specified class.

  • If you return a String that starts with "redirect:" then Spring MVC does the appropriate response.setRedirect(...) call for you.

    @Controller public class MyFavoritePatientsController { ... @RequestMapping("module/jspexample/addFavoritePatient") public String addFavoritePatient(@RequestParam("patient_id") Integer patientId) { Patient p = Context.getPatientService().getPatient(patientId); if (p != null) MyFavoritePatientUtil.addFavoritePatient(Context.getAuthenticatedUser(), p); return "redirect:myfavoritepatients.form"; } }

There's a nicer way to do this, though. OpenMRS provides a bunch of editor classes that know how to convert between a String (in an HTTP request) and our domain objects. In this case org.openmrs.propertyeditor.PatientEditor.

By default, Spring will only automatically convert a small number of parameter types that it knows: Integers, Doubles, Booleans, and a few more. In our controller we need to tell it about other types it should automatically convert:

@Controller public class MyFavoritePatientsController { @InitBinder public void initBinder(WebDataBinder wdb) { wdb.registerCustomEditor(Patient.class, new PatientEditor()); } ... @RequestMapping("module/jspexample/addFavoritePatient") public String addFavoritePatient(@RequestParam(value="patient_id", required=false) Patient patient) { if (patient != null) MyFavoritePatientUtil.addFavoritePatient(Context.getAuthenticatedUser(), patient); return "redirect:myfavoritepatients.form"; } }

MyFavoritePatientsUtil

public class MyFavoritePatientsUtil { private static String cohortName(User u) { return "Favorite Patients for User " + u.getUserId(); } private static List<Patient> getPatients(Cohort cohort) { if (cohort == null) return new ArrayList<Patient>(); else return Context.getPatientSetService().getPatients(cohort.getMemberIds()); } public static List<Patient> getFavoritePatients(User u) { Cohort c = Context.getCohortService().getCohort(cohortName(u)); return getPatients(c); } public static void addFavoritePatient(User u, Patient p) { Cohort c = Context.getCohortService().getCohort(cohortName(u)); if (c == null) { c = new Cohort(); c.setName(cohortName(u)); c.setDescription("Automatically generated"); } c.addMember(p.getPatientId()); Context.getCohortService().saveCohort(c); } }