Package org.olat.core.gui.control.generic.portal

Source Code of org.olat.core.gui.control.generic.portal.PortalImpl

/**
* OLAT - Online Learning and Training<br>
* http://www.olat.org
* <p>
* Licensed under the Apache License, Version 2.0 (the "License"); <br>
* you may not use this file except in compliance with the License.<br>
* You may obtain a copy of the License at
* <p>
* http://www.apache.org/licenses/LICENSE-2.0
* <p>
* Unless required by applicable law or agreed to in writing,<br>
* software distributed under the License is distributed on an "AS IS" BASIS, <br>
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br>
* See the License for the specific language governing permissions and <br>
* limitations under the License.
* <p>
* Copyright (c) 1999-2006 at Multimedia- & E-Learning Services (MELS),<br>
* University of Zurich, Switzerland.
* <p>
*/

package org.olat.core.gui.control.generic.portal;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.olat.core.gui.UserRequest;
import org.olat.core.gui.components.Component;
import org.olat.core.gui.components.link.Link;
import org.olat.core.gui.components.link.LinkFactory;
import org.olat.core.gui.components.velocity.VelocityContainer;
import org.olat.core.gui.control.Controller;
import org.olat.core.gui.control.ControllerEventListener;
import org.olat.core.gui.control.DefaultController;
import org.olat.core.gui.control.Event;
import org.olat.core.gui.control.WindowControl;
import org.olat.core.gui.control.generic.portal.Portal;
import org.olat.core.gui.control.generic.portal.PortalImpl;
import org.olat.core.gui.control.generic.portal.PortletContainer;
import org.olat.core.gui.control.generic.portal.PortletFactory;
import org.olat.core.gui.translator.PackageTranslator;
import org.olat.core.gui.translator.Translator;
import org.olat.core.logging.AssertException;
import org.olat.core.logging.Tracing;
import org.olat.core.util.ObjectCloner;
import org.olat.core.util.Util;
import org.olat.core.util.prefs.Preferences;

/**
* Description:<br>
* The portal implementation has the ability to display a portal page as defined in the
* WEB-INF/olat_extensions.xml. Use the PortalFactory to create a new portal instance.
*
* <P>
* Initial Date:  08.07.2005 <br>
* @author gnaegi
*/
public class PortalImpl extends DefaultController implements Portal, ControllerEventListener {
  private static final String VELOCITY_ROOT = Util.getPackageVelocityRoot(PortalImpl.class);

  private static String MODE_EDIT = "editMode";
 
  private VelocityContainer portalVC;
  private Translator trans;
  private List<List<String>> portalColumns; // list containing the lists of portlets
  private Map<String, PortletContainer> portletContainers; // map of all portlet containers (defined in portal columns + inactive portlets)
  private List<String> inactivePortlets; // list containing the names of inactive portlets
  private String name;
  private Map<String, Map<String, String>> portletsConfigurations;

  /**
   * Do use PortalFactory for create new Portals
   */
  public PortalImpl(){
    super(null);
    // used by spring framework. Use PortalFactory to create a runtime portal
  }
 
  /**
   * Do use PortalFactory for create new Portals!
   * @param portalName identifyer of the portal
   * @param ureq
   * @param wControl
   * @param portalColumns List containing the default columns and rows
   * @param portletsConfigurations Map containing the portlet configurations
   */
  protected PortalImpl(String portalName, UserRequest ureq, WindowControl wControl, List<List<String>> portalColumns, Map<String, Map<String, String>> portletsConfigurations) {
    super(wControl);
    this.name = portalName;
   
    this.portalColumns = portalColumns;
    this.portletsConfigurations = portletsConfigurations;   
    this.portletContainers = new HashMap<String, PortletContainer>();
    this.inactivePortlets = new ArrayList<String>();

    this.trans = new PackageTranslator(Util.getPackageName(PortalImpl.class), ureq.getLocale());
    this.portalVC = new VelocityContainer("portalVC", VELOCITY_ROOT + "/portal.html", trans, this);    // initialize arrays

    // calculate the column css classes based on YAML schema
    int cols = portalColumns.size();
    List<String> columnCssClassWrapper = new ArrayList(cols);
    columnCssClassWrapper.add(0, ""); // empty, in velocity things start with 1...
    List<String> columnCssClassInner = new ArrayList(cols);
    columnCssClassInner.add(0, ""); // empty, in velocity things start with 1...
    switch (cols) {
    case 0:
      // do nothing
      break;
    case 1:
      // empty css class
      columnCssClassWrapper.add(1, "");
      columnCssClassInner.add(1, "");
      break;
    case 2:
      // 50% each
      columnCssClassWrapper.add(1, "b_c50l");
      columnCssClassInner.add(1, "b_subcl");
      columnCssClassWrapper.add(2, "b_c50r");
      columnCssClassInner.add(2, "b_subcr");
      break;
    case 3:
      // 33% each
      columnCssClassWrapper.add(1, "b_c33l");
      columnCssClassInner.add(1, "b_subcl");
      columnCssClassWrapper.add(2, "b_c33l");
      columnCssClassInner.add(2, "b_subcl");
      columnCssClassWrapper.add(3, "b_c33r");
      columnCssClassInner.add(3, "b_subcr");
      break;
    case 4:
      // 25% each
      columnCssClassWrapper.add(1, "b_c25l");
      columnCssClassInner.add(1, "b_subcl");
      columnCssClassWrapper.add(2, "b_c25l");
      columnCssClassInner.add(2, "b_subcl");
      columnCssClassWrapper.add(3, "b_c25l");
      columnCssClassInner.add(3, "b_subcl");
      columnCssClassWrapper.add(4, "b_c25r");
      columnCssClassInner.add(4, "b_subcr");
      break;
    default:
      // do log as error but don't make redscreen for user.
      Tracing.logError("only up to 4 portal columns supported but " + cols + " columns available in portal::" + portalName, this.getClass());
      break;
    }
    portalVC.contextPut("columnCssClassWrapper", columnCssClassWrapper);
    portalVC.contextPut("columnCssClassInner", columnCssClassInner);
   
    // init all portlets enabled in the portal columns
    initPortlets(ureq);
    // push the columns to velocity
    this.portalVC.contextPut("portalColumns", portalColumns);
    // push list of inactive portlets to velocity
    this.portalVC.contextPut("inactivePortlets", inactivePortlets);
    // push all portlets to velocity
    this.portalVC.contextPut("portletContainers", portletContainers);
    this.portalVC.contextPut("locale", ureq.getLocale());
    // in run mode
    this.portalVC.contextPut(MODE_EDIT, Boolean.FALSE);
    setInitialComponent(portalVC);
  }

  /**
   * Factory method to create a portal instance of the current type. Used by the PortalFactory.
   * The method checks for invalid user configurations and removes them
   * @param wControl
   * @param ureq
   * @return PortalImpl of same type and configuration
   */
  protected PortalImpl createInstance(WindowControl wContr, UserRequest ureq) {
    // user users personal configuration
    List userColumns = getUserPortalColumns(ureq);
    // clone default configuration for this user if user has no own configuration
    if (userColumns == null) {
      userColumns = (List) ObjectCloner.deepCopy(this.portalColumns);
    }
    // always use a copy of the original configuration
    Map<String, Map<String, String>> configurations = (Map) ObjectCloner.deepCopy(this.portletsConfigurations);

    // check if users portal columns contain only defined portals. remove all non existing portals
    // to make it possible to change the portlets in a next release or to remove a portlet
    List<List<String>> cleanedUserColumns = new ArrayList<List<String>>();
    Set availablePortlets = configurations.keySet();
    Iterator colIter = userColumns.iterator();
    while (colIter.hasNext()) {
      // add this row as new cleaned row to columns
      List<String> cleanedRow = new ArrayList<String>();
      cleanedUserColumns.add(cleanedRow);
      // check all portlets in old row and copy to cleaned row if it exists
      List row = (List) colIter.next();
      Iterator rowIter = row.iterator();
      while (rowIter.hasNext()) {
        String portletName = (String) rowIter.next();
        if (availablePortlets.contains(portletName)) {
          cleanedRow.add(portletName);         
        }
        // discard invalid portlet names
      }
    }
    return new PortalImpl(this.name, ureq, wContr, cleanedUserColumns, configurations);
  }
 
  private List getUserPortalColumns(UserRequest ureq) {
    Preferences gp = ureq.getUserSession().getGuiPreferences();
    return (List) gp.get(PortalImpl.class, "userPortalColumns" + name);
  }

  private void saveUserPortalColumnsConfiguration(UserRequest ureq, List userColumns) {
    Preferences gp = ureq.getUserSession().getGuiPreferences();
    gp.putAndSave(PortalImpl.class, "userPortalColumns" + name, userColumns);
  }
 
  /** 
   * Initialize all portles found in the configuration
   * @param ureq
   */
  private void initPortlets(UserRequest ureq) {
    // load all possible portlets, portlets run controller is only loaded when really used
    Iterator portletsIter = this.portletsConfigurations.keySet().iterator();
    while (portletsIter.hasNext()) {
      String portletName = (String) portletsIter.next();
      Map<String, String> portletConfiguration = this.portletsConfigurations.get(portletName);
      if (portletConfiguration == null)
        throw new AssertException("No portlet configuration found for portlet::" + portletName + " in portal::" + this.name);
      // add portal and portlet name to configuration
      portletConfiguration.put("portalName", this.name);
      portletConfiguration.put("id", portletName);
      PortletContainer pc = PortletFactory.getPortletContainerFor(portletConfiguration, getWindowControl(), ureq);     
      pc.addControllerListener(this);
      // remember this portlet container
      this.portletContainers.put(portletName, pc);
      String addLinkName = "command.add." + portletName;
      Link tmp = LinkFactory.createCustomLink(addLinkName, addLinkName, "add", Link.BUTTON_XSMALL, portalVC, this);
      tmp.setUserObject(portletName);
      // and add to velocity
      this.portalVC.put(portletName, pc.getInitialComponent());
     
      // check if portlet is active for this user
      Iterator colIter = this.portalColumns.iterator();
      boolean isActive = false;
      while (colIter.hasNext()) {
        List row = (List) colIter.next();
        Iterator rowIter = row.iterator();
        while (rowIter.hasNext()) {
          String activePortletName = (String) rowIter.next();
          if (portletName.equals(activePortletName)) isActive = true;
        }
      }
      if (isActive) {
        // initialize portlet container for active portlets only
        pc.initializeRunComponent(ureq);
      } else {
        // add it to inacitve portlets list if not active
        inactivePortlets.add(portletName);
      }
    }
    // update links on visible portlets
    updatePositionChangeLinks();
  }
 
  /**
   * @see org.olat.core.gui.control.DefaultController#event(org.olat.core.gui.UserRequest, org.olat.core.gui.components.Component, org.olat.core.gui.control.Event)
   */
  public void event(UserRequest ureq, Component source, Event event) {
    if (source instanceof Link && portalVC.getComponents().containsValue(source)) {
      Link tmp = (Link)source;
      String portletName = (String)tmp.getUserObject();
      List<String> firstColumn = this.portalColumns.get(0);
      PortletContainer pc = this.portletContainers.get(portletName);
      if (pc == null) throw new AssertException("trying to add portlet with name::" + portletName + " to portal, but portlet container did not exist. Could be a user modifying the URL...");
      // add to users portlet list
      firstColumn.add(portletName);
      // remove from inactive portlets list
      this.inactivePortlets.remove(portletName);
      // initialize portlet run component
      pc.initializeRunComponent(ureq);
      // save user config in db
      saveUserPortalColumnsConfiguration(ureq, portalColumns);
      // update possible links in gui
      updatePositionChangeLinks();
      portalVC.setDirty(true);
    }
    // nothin to catch
  }

  /**
   * Enable/disable the edit mode of the portal
   * @param editModeEnabled true: enabled, false: disabled
   */
  public void setIsEditMode(UserRequest ureq, Boolean editModeEnabled) {
    updatePorletContainerEditMode(ureq, editModeEnabled);
    this.portalVC.contextPut(MODE_EDIT, editModeEnabled);
  }
 
  /**
   * Updates all portles using the given mode
   * @param editMode true: edit mode activated, false: deactivated
   */
  private void updatePorletContainerEditMode(UserRequest ureq, Boolean editMode) {
    Iterator portletsIter = this.portletsConfigurations.keySet().iterator();
    while (portletsIter.hasNext()) {
      String portletName = (String) portletsIter.next();
      PortletContainer pc = this.portletContainers.get(portletName);
      pc.setIsEditMode(ureq, editMode);
    }
  }
 
  /**
   * @see org.olat.core.gui.control.DefaultController#event(org.olat.core.gui.UserRequest, org.olat.core.gui.control.Controller, org.olat.core.gui.control.Event)
   */
  public void event(UserRequest ureq, Controller source, Event event) {
    if (source instanceof PortletContainer) {
      PortletContainer pc = (PortletContainer) source;
      String cmd = event.getCommand();
      boolean found = false;
      for (int column = 0; column < portalColumns.size(); column++) {
        List rows = portalColumns.get(column);
        for (int row = 0; row < rows.size(); row++) {
          String portletName = (String) rows.get(row);
          if (portletName.equals(pc.getPortlet().getConfiguration().get("id"))){
            if (cmd.equals("move.up")) {
              Collections.swap(rows, row, row - 1);
              found = true;
              break;
            } else if (cmd.equals("move.down")){
              Collections.swap(rows, row, row+1);
              found = true;
              break;
            } else if (cmd.equals("move.right")){
              rows.remove(row);
              List<String> newCol = portalColumns.get(column + 1);
              newCol.add(portletName);
              found = true;
              break;
            } else if (cmd.equals("move.left")){
              rows.remove(row);
              List<String> newCol = portalColumns.get(column - 1);
              newCol.add(portletName);
              found = true;
              break;
            } else if (cmd.equals("close")){
              pc.deactivateRunComponent();
              rows.remove(row);
              this.inactivePortlets.add(portletName);
              found = true;
              break;
            }
          }
        }
        if (found) break;
      }
      // save user config in db
      saveUserPortalColumnsConfiguration(ureq, portalColumns);
      // update possible links in gui
      updatePositionChangeLinks();
      portalVC.setDirty(true);
    }
  }

  /**
   * Updates the velocity containers of all portlet containers to display the move
   * links correctly
   */
  private void updatePositionChangeLinks(){
    Iterator colIter = portalColumns.iterator();
    int colcount = 0;
    while (colIter.hasNext()) {
      List rows = (List) colIter.next();
      Iterator rowIter = rows.iterator();
      int rowcount = 0;
      while (rowIter.hasNext()) {
        String portletName = (String) rowIter.next();
        PortletContainer pc = this.portletContainers.get(portletName);
        // up command
        if(rowcount == 0) pc.setCanMoveUp(false);
        else pc.setCanMoveUp(true);
        // down command
        if (rowIter.hasNext()) pc.setCanMoveDown(true);
        else pc.setCanMoveDown(false);
        // left command
        if(colcount == 0) pc.setCanMoveLeft(false);
        else pc.setCanMoveLeft(true);
        // right command
        if (colIter.hasNext()) pc.setCanMoveRight(true);
        else pc.setCanMoveRight(false);
       
        rowcount++;
      }
      colcount++;
    }
  }
 
  /**
   * @see org.olat.core.gui.control.DefaultController#doDispose(boolean)
   */
  protected void doDispose() {
    // cleanup all portlet containers
    Iterator iter = portletContainers.values().iterator();
    while (iter.hasNext()) {
      PortletContainer element = (PortletContainer) iter.next();
      element.dispose();
    }
    portletContainers = null;
  }

  /**
   * @see org.olat.gui.control.generic.portal.Portal#setPortalColumns(java.util.List)
   */
  public void setPortalColumns(List<List<String>> portalColumns) {
    this.portalColumns = portalColumns;
  }

  /**
   * Bean method used by spring
   * @param numbColumns
   */
  public void setPortletsConfigurations(Map<String, Map<String, String>> config) {
    this.portletsConfigurations = config;
  }

  /**
   * Bean method used by spring
   * @param numbColumns
   */
  public void setName(String name){
    this.name = name;
  }

  /**
   * @return Name of portal
   */
  public String getName(){
    return this.name;
  }

}
TOP

Related Classes of org.olat.core.gui.control.generic.portal.PortalImpl

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.