Implementing an Item Selector

Implementing an Item Selector

Item selectors are pop-up dialogs for selecting assets, such as documents, videos, or users.

By configuring the Item Selector’s criteria and defining its usage, you can create Item Selector dialogs for your own application.

Item selectors pop up so users can select assets.

Here you’ll learn how to create an Item Selector.

Start with a Sample Module

To implement an Item Selector, you must embed it in an application, such as a module for a widget. This example uses an MVC Portlet with a JSP view. The Item Selector displays a list of Roles to be selected.

  1. Download the sample module:

    curl https://resources.learn.liferay.com/dxp/latest/en/building-applications/data-frameworks/liferay-f5d5.zip -O
    
    unzip liferay-f5d5.zip
    
  2. Start a Liferay DXP Docker container with the following command:

    docker run -it -m 8g -p 8080:8080 --name lrdev liferay/portal:7.3.2-ga3
    
  3. Run the following commands from the root of the module to build and deploy to your Docker container:

    ./gradlew deploy -Ddeploy.docker.container.id=$(docker ps -lq)
    
    tip

    This command is the same as copying the deployed jars to /opt/liferay/osgi/modules on the Docker container.

  4. Confirm the deployment in the Liferay Docker container console.

    STARTED com.acme.f5d5.web_1.0.0 [1017]
    

    The example portlet module is deployed. When you add it to a page, it’s a simple portlet with one button:

    The portlet has one button that opens the Item Selector.

  5. Click the Select button and the Item Selector appears:

    The Item Selector shows items that can be selected by checking the box.

  6. Select an item and that item’s value appears in a JavaScript alert box. Since this Item Selector selects Roles, the value is the primary key of the Role selected.

Now you can see how this works.

Setting an Item Selector’s Criteria in your Controller

Open the F5D5MVCPortlet.java class. In an MVC Portlet, the portlet class is the controller class (the C in MVC). It must do two things:

  • Define the necessary criteria for the selector (i.e., what entity does it select?)
  • Create a URL for that criteria
  1. At the bottom of the class, OSGi injects an ItemSelector class instance because of the @Reference annotation.

    @Reference
    private ItemSelector _itemSelector;
    
  2. Scroll to the portlet’s render method.

  3. A criterion class to represent the desired entities to display in the item selector is created. Criterion classes must implement the ItemSelectorCriterion interface.

    This example uses a reference to RoleItemSelectorCriterion so that Roles are shown in the selector. It’s defined by creating a new instance of the class:

    ItemSelectorCriterion itemSelectorCriterion =
         new RoleItemSelectorCriterion();
    
    tip

    If no criterion exists for the type of entity that you need, you can define your own ItemSelectorCriterion by extending [BaseItemSelectorCriterion]).

  4. Next, you need a return type class to represent the information the entities provided when users select them. Return type classes must implement the ItemSelectorReturnType interface. For example, a return type class may be used to return the entity’s URL, UUID, or primary key. The return type class is added to the criterion class created previously.

    important

    Every criterion used must have at least one return type associated with it.

    This example uses a reference to UUIDItemSelectorReturnType to define the selected Roles’ UUID value as the crucial data to return. If multiple Roles are selected, they are returned as a comma-delimited list.

    note

    If a UUID is not available, the primary key is returned.

  5. Define the return type by registering it with the item criterion:

    	itemSelectorCriterion.setDesiredItemSelectorReturnTypes(
    		new UUIDItemSelectorReturnType());
    
    tip

    If no return type exists for the type of information that you need, then you can define your own ItemSelectorReturnType implementation.

    The item selector uses these two classes to decide what selection views of items (presented as tabs) to show and how to identify each item.

  6. Now you can use the criteria to generate a URL for the Item Selector. This URL creates the Item Selector dialog in your front-end code.

    The RequestBackedPortletURLFactory class can quickly generate an item selector URL using the criteria:

    	PortletURL itemSelectorURL = _itemSelector.getItemSelectorURL(
    		RequestBackedPortletURLFactoryUtil.create(renderRequest),
    		renderResponse.getNamespace() + "selectRole",
    		itemSelectorCriterion);
    
    important

    The String you use to generate the URL (in this example, selectRole) is the dialog’s event name. This must match a value you’ll use later when creating the dialog in your front-end code.

  7. Add the item selector URL to the renderRequest so that it’s available in the JSP:

    renderRequest.setAttribute(F5D5WebKeys.ITEM_SELECTOR_URL, itemSelectorURL);
    

    The view.jsp file is where the front-end code is defined. The renderRequest object in your Java class’s render method is passed to the JSP file later. Use a constant to make sure you identify the URL consistently in both the controller (portlet class) and the view (JSP).

  8. Finally, call MVCPortlet’s render method to continue the rendering process once your code is executed:

    super.render(renderRequest, renderResponse);
    

That’s the controller code. Execution now passes to the view layer (the V in MVC), which is implemented in the view.jsp file.

Use the Item Selector in Your View

  1. You must retrieve the item selector and define a way to use it in your front-end code Open view.jsp from the example.

    <%
       String itemSelectorURL = String.valueOf(request.getAttribute(F5D5WebKeys.ITEM_SELECTOR_URL));
    %>
    

    Once you have the URL, you must provide a way to open the item selector and then define how to use the result.

  2. You can use a Clay button tag to create a button to open your item selector:

    <clay:button
      id='<%= liferayPortletResponse.getNamespace() + "selectRoleButton" %>'
      label="Select"
    />
    

    The clay:button tag creates a button (with the ID selectRoleButton and the label Select displayed on the screen) on your widget. This button can be identified by the String <portlet:namespace />selectRoleButton.

  3. Use the <script> tag to embed JavaScript that opens the item selector:

    <script>
       var selectRoleButton = document.getElementById('<portlet:namespace />selectRoleButton');
    
       selectRoleButton.addEventListener(
          'click',
          function(event) {
             new Liferay.Util.openSelectionModal(
                {
                   onSelect: function (event) {
                      alert(event.value);
                   },
                   selectEventName: '<portlet:namespace />selectRole',
                   title: 'Select Role',
                   url: '<%= request.getAttribute(F5D5WebKeys.ITEM_SELECTOR_URL) %>'
                }
             );
          }
       );
    </script>
    

This snippet of JavaScript first retrieves the Select Role button through its identifier (portlet:namespace />selectRoleButton). Then it adds an event listener to create the item selector dialog when clicked.

The Liferay.Util.openSelectionModal method creates the dialog.

The onSelect field must define a function to handle the value when it is clicked. Define the dialog’s behavior when the user makes a selection within this function. This implementation shows an alert box containing the selected value.

The value for the eventName field must match the String you used with the RequestBackedPortletURLFactory in the Java code (in this example, selectRole). You must also retrieve the item selector URL from the request where the controller stored it, using the same constant to identify it, supplying it in the url field.

tip

If you want your item selector to support selecting multiple items, you can enable multiple selection by adding multiple: true to the openSelectionModal call.

Use the item selection, stored in the event. The data type and information contained in the result depends on what return type class you used in the Java code. Since this example uses UUIDItemSelectorReturnType, the data is a String value with the UUIDs of one or more selected items.

Inside the selection function, you implement how you want to use the value. This example shows a simple JavaScript alert.

Now that you’ve seen how it works, you can replace the JavaScript alert with something more useful.

Add A Form

When you use an item selector, you want the selected value inserted into a form. Here’s how to do that:

  1. Open view.jsp.

  2. Find the <clay:button> tag. Surround it with a form like this:

    <form name="<portlet:namespace/>fm">
       <input name="role" />
       <clay:button
          id='<%= liferayPortletResponse.getNamespace() + "selectRoleButton" %>'
          label="Select"
       />
    </form>
    

    Now you have a form with one input field called role.

  3. Scroll down to the JavaScript alert. Replace the JavaScript alert with a call to Liferay’s front-end setFormValues method:

    Liferay.Util.setFormValues(document.<portlet:namespace />fm, {
           role: event.value
    });
    
  4. Redeploy the portlet:

    ./gradlew deploy -Ddeploy.docker.container.id=$(docker ps -lq)
    
  5. Now select an item as you did before. Its ID is inserted into the form field you created.

Conclusion

Congratulations! You now know how to implement an item selector!