Package devplugin

Source Code of devplugin.PluginTreeNode$RemovedProgramsHandler

/*
* TV-Browser
* Copyright (C) 04-2003 Martin Oberhauser (martin@tvbrowser.org)
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
*
* CVS information:
*  $RCSfile$
*   $Source$
*     $Date: 2010-10-27 19:49:08 +0200 (Wed, 27 Oct 2010) $
*   $Author: bananeweizen $
* $Revision: 6793 $
*/
package devplugin;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

import javax.swing.Action;

import tvbrowser.core.TvDataUpdateListener;
import tvbrowser.core.TvDataUpdater;
import tvbrowser.ui.pluginview.Node;
import tvbrowser.ui.pluginview.PluginTreeModel;
import util.program.ProgramUtilities;
import util.ui.Localizer;
import util.ui.UIThreadRunner;


/**
* The PluginTreeNode class represents a single node of the plugin tree view.
*
*/
public class PluginTreeNode implements Comparable<PluginTreeNode> {

  private static final util.ui.Localizer mLocalizer =
      util.ui.Localizer.getLocalizerFor(PluginTreeNode.class);

  private byte mNodeType;
  private ArrayList<PluginTreeNode> mChildNodes;
  private Object mObject;
  private ArrayList<PluginTreeListener> mNodeListeners;
  private Marker mMarker;
  private Node mDefaultNode;
  private boolean mGroupingByDate;
  private boolean mGroupWeekly;

  /**
   * cache todays date for update of all tree nodes
   */
  private static Date mNodeToday;
  /**
   * cache tomorrows date for update of all tree nodes
   */
  private static Date mNodeTomorrow;
  /**
   * cache yesterdays date for update of all tree nodes
   */
  private static Date mNodeYesterday;

  private PluginTreeNode(final byte type, final Object object) {
    mChildNodes = null; // do not initialize to save memory
    mNodeType = type;
    mObject = object;
    mDefaultNode = new Node(type, mObject);
    mNodeListeners = null; // do not initialize to save memory
    mGroupingByDate = true;
    mGroupWeekly = false;
  }

  private PluginTreeNode(final int type, final Object object) {
    this((byte)type, object);
  }

  /**
   * Creates a new PluginTreeNode object with a specified title
   * @param title
   */
  public PluginTreeNode(final String title) {
    this(Node.CUSTOM_NODE, title);
  }

  /**
   * Creates a new root PluginTreeNode
   * On TV listings updates, the {@link PluginTreeListener} gets informed.
   * @param marker
   */
  public PluginTreeNode(final Marker marker) {
    this(marker, true);
  }

  /**
   * creates a plugin root node WITHOUT marker
   *
   * @since 3.0
   * @param plugin
   *
   */
  public PluginTreeNode(final Plugin plugin) {
    this(true, plugin);
  }

  /**
   * Creates a new root PluginTreeNode
   * @param plugin
   * @param handleTvDataUpdate specifies, if the {@link PluginTreeListener}
   * should get called on TV listings updates
   * @since 3.0
   */
  public PluginTreeNode(final boolean handleTvDataUpdate, final Plugin plugin) {
    this(Node.PLUGIN_ROOT, plugin);

    if (handleTvDataUpdate) {
      addRemovedProgramsListener();
    }
  }

  /**
   * Creates a new root PluginTreeNode
   * @param marker
   * @param handleTvDataUpdate specifies, if the {@link PluginTreeListener}
   * should get called on TV listings updates
   */
  public PluginTreeNode(final Marker marker, final boolean handleTvDataUpdate) {
    this(Node.PLUGIN_ROOT, marker);
    mMarker = marker;

    if (handleTvDataUpdate) {
      addRemovedProgramsListener();
    }

  }

  private void addRemovedProgramsListener() {
    final RemovedProgramsHandler removedProgramsHandler = new RemovedProgramsHandler();

    TvDataUpdater.getInstance().addTvDataUpdateListener(new TvDataUpdateListener() {
      public void tvDataUpdateStarted() {
        removedProgramsHandler.clear();
      }

      public void tvDataUpdateFinished() {
        refreshAllPrograms(removedProgramsHandler);
        update();
        final Program[] removedPrograms = removedProgramsHandler
                .getRemovedPrograms();
        fireProgramsRemoved(removedPrograms);
      }
    });
  }

  /**
   * Creates a new Node containing a program item.
   * @param item
   */
  public PluginTreeNode(final ProgramItem item) {
    this(Node.PROGRAM, item);
    mDefaultNode.setAllowsChildren(false);
  }


  public void addNodeListener(final PluginTreeListener listener) {
    if (mNodeListeners == null) {
      mNodeListeners = new ArrayList<PluginTreeListener>(1);
    }
    mNodeListeners.add(listener);
  }

  public boolean removeNodeListener(final PluginTreeListener listener) {
    if (mNodeListeners == null) {
      return false;
    }
    return mNodeListeners.remove(listener);
  }

  public void removeAllNodeListeners() {
    if (mNodeListeners == null) {
      return;
    }
    mNodeListeners.clear();
    mNodeListeners = null;
  }

  /**
   * Remove all programs from this node which are not available any more
   */
  private synchronized void refreshAllPrograms(RemovedProgramsHandler handler) {
    // non initialized child collection, if it is empty
    if (mChildNodes == null) {
      return;
    }
    for (int i=mChildNodes.size()-1; i>=0; i--) {
      final PluginTreeNode node = mChildNodes.get(i);
      node.mMarker = mMarker;

      if (node.isLeaf()) {
        final ProgramItem progItemInTree = (ProgramItem) node.getUserObject();
        final Program progInTree = progItemInTree.getProgram();

        if (progInTree == null) {
          node.removeProgram(progItemInTree);
        }
        else {
          if (progInTree.getProgramState() == Program.WAS_DELETED_STATE) {
            removeProgram(progInTree);
            handler.addRemovedProgram(progInTree);
          } else if (progInTree.getProgramState() == Program.WAS_UPDATED_STATE) {
            final Program updatedProg = Plugin.getPluginManager().getProgram(
                progInTree.getDate(), progInTree.getID());
            progItemInTree.setProgram(updatedProg);
          }
        }
      }
      else {
        node.refreshAllPrograms(handler);
      }
    }
  }

  private void fireProgramsRemoved(Program[] progArr) {
    if (mNodeListeners == null) {
      return;
    }
    for (int i=0; i<mNodeListeners.size(); i++) {
      PluginTreeListener listener = mNodeListeners.get(i);
      listener.programsRemoved(progArr);
    }
  }

  public Node getMutableTreeNode() {
    return mDefaultNode;
  }

  /**
   * Adds a an action menu to this node
   * @param menu
   */
  public void addActionMenu(final ActionMenu menu) {
    mDefaultNode.addActionMenu(menu);
  }


  public void removeAllActions() {
    mDefaultNode.removeAllActionMenus();
  }

  public void addAction(Action action) {
    addActionMenu(new ActionMenu(action));
  }

  public ActionMenu[] getActionMenus() {
    return mDefaultNode.getActionMenus();
  }

  /**
   * Sets the formatter for this node and all of the child nodes.
   * @param formatter the formatter
   */
  public void setNodeFormatter(NodeFormatter formatter) {
    mDefaultNode.setNodeFormatter(formatter);
  }

  /**
   * Enables/Disables the 'grouping by date'-feature.
   * Default is 'enabled'
   *
   * @param enable
   */
  public void setGroupingByDateEnabled(boolean enable) {
    mGroupingByDate = enable;
  }

  /**
   * Enables/Disables 'grouping by week' for nodes showing
   * programs by date. Only evaluated if "grouping by date"
   * is enabled.
   * Default is 'disabled'
   *
   * @see #setGroupingByDateEnabled(boolean)
   *
   * @param enable
   */
  public void setGroupingByWeekEnabled(final boolean enable) {
    mGroupWeekly = enable;
  }

  private NodeFormatter getNodeFormatter(final boolean isWeekNodesEnabled) {
    return mDefaultNode.getNodeFormatter(isWeekNodesEnabled);
  }

  private void createDefaultNodes() {
    mDefaultNode.removeAllChildren();
    // non initialized child collection, if it is empty
    if (mChildNodes == null) {
      return;
    }
    PluginTreeNode[] items = mChildNodes.toArray(new PluginTreeNode[mChildNodes.size()]);
    Arrays.sort(items);
    Date currentDate = null;
    for (PluginTreeNode n : items) {
      if (!n.isLeaf()) {
        if (n.mGroupingByDate) {
          n.createDateNodes();
        }
        else {
          n.createDefaultNodes();
        }
        mDefaultNode.add(n.getMutableTreeNode());
      }
      else {
        if (n.mNodeType == Node.PROGRAM) {
          ProgramItem progItem = (ProgramItem)n.getUserObject();
          Node node = n.getMutableTreeNode();

          if (currentDate == null) {
            currentDate = Date.getCurrentDate();
          }
          if(progItem.getProgram().getDate().addDays(1).compareTo(currentDate) >= 0) {
            mDefaultNode.add(node);
          }
        }
        else {
          mDefaultNode.add(n.getMutableTreeNode());
        }
      }
    }
  }

  private void createDateNodes() {
    /* We create new folders for each day and assign the program items
       to the appropriate folder */

    mDefaultNode.removeAllChildren();

    // return if no nodes available
    if (mChildNodes == null || mChildNodes.size() == 0) {
      return;
    }

    Map<Date, ArrayList<PluginTreeNode>> dateMap = new HashMap<Date, ArrayList<PluginTreeNode>>()// key: date; value: ArrayList of ProgramItem objects
    Iterator<PluginTreeNode> it = mChildNodes.iterator();
    while (it.hasNext()) {
      PluginTreeNode n = it.next();
      if (!n.isLeaf()) {
        if (n.mGroupingByDate) {
          n.createDateNodes();
        }
        else {
          n.createDefaultNodes();
        }
        mDefaultNode.add(n.getMutableTreeNode());
      }
      else {
        Date date = ((ProgramItem) n.getUserObject()).getDate();

        if(date.compareTo(mNodeYesterday) >= 0) {
          ArrayList<PluginTreeNode> list = dateMap.get(date);
          if (list == null) {
            list = new ArrayList<PluginTreeNode>();
            dateMap.put(date, list);
          }

          list.add(n);
        }
      }
    }

    // Create the new nodes
    Set<Date> keySet = dateMap.keySet();
    Date[] dates = new Date[keySet.size()];
    keySet.toArray(dates);
    Arrays.sort(dates);
    Node node=null;
    String lastDateStr="";
    int numPrograms = 0;
    for (Date date : dates) {
      numPrograms += dateMap.get(date).size();
    }
    // show week nodes if there are less than 2 programs per day on average
    boolean createWeekNodes = mGroupWeekly && (numPrograms <= dates.length * 2);
    boolean isShowingWeekNodes = createWeekNodes;

    for (Date date : dates) {
      String dateStr;
      if (mNodeYesterday.equals(date)) {
        dateStr = Localizer.getLocalization(Localizer.I18N_YESTERDAY);
        isShowingWeekNodes = false;
      }
      else if (mNodeToday.equals(date)) {
        dateStr = Localizer.getLocalization(Localizer.I18N_TODAY);
        isShowingWeekNodes = false;
      }
      else if (mNodeTomorrow.equals(date)) {
        dateStr = Localizer.getLocalization(Localizer.I18N_TOMORROW);
        isShowingWeekNodes = false;
      }
      else {
        if (createWeekNodes) {
          int weeks = date.getNumberOfDaysSince(mNodeToday) / 7;
          if (weeks <= 3) {
            dateStr = mLocalizer.msg("weeks."+weeks,"in {0} weeks",weeks);
          }
          else {
            dateStr = mLocalizer.msg("weeks.later","later");
          }
        }
        else {
          dateStr = date.getLongDateString();
        }
      }
      if (!dateStr.equals(lastDateStr)) {
        node = new Node(Node.STRUCTURE_NODE, dateStr);
        mDefaultNode.add(node);
        lastDateStr = dateStr;
      }
      List<PluginTreeNode> list = dateMap.get(date);
      PluginTreeNode[] nodeArr = new PluginTreeNode[list.size()];
      list.toArray(nodeArr);
      Arrays.sort(nodeArr);
      for (PluginTreeNode element : nodeArr) {
        Node newNode = new Node((ProgramItem)element.getUserObject());
        newNode.setNodeFormatter(element.getNodeFormatter(createWeekNodes && isShowingWeekNodes));
        node.add(newNode);
      }

      isShowingWeekNodes = createWeekNodes;
    }
  }

  public Object getUserObject() {
    return mObject;
  }

  public synchronized void removeAllChildren() {
    if (mMarker != null) {
      Program[] programs = getPrograms();
      for (Program program : programs) {
        program.unmark(mMarker);
      }
    }
    if (mChildNodes != null) {
      mChildNodes.clear();
      mChildNodes = null;
    }
    mDefaultNode.removeAllChildren();
  }


  public synchronized void add(final PluginTreeNode node) {
    // create collection on demand only
    if (mChildNodes == null) {
      mChildNodes = new ArrayList<PluginTreeNode>(1);
    }
    mChildNodes.add(node);
    node.mMarker = mMarker;
  }

  public boolean contains(final Program prog, final boolean recursive) {
    PluginTreeNode node = findProgramTreeNode(prog, recursive);
    return node != null;
  }

  public boolean contains(final Program prog) {
    return contains(prog, false);
  }

  /**
   * Refreshes the tree in the user interface. Call this method after you have added/removed/changed
   * nodes of the tree or its children. Otherwise those changes will not get visible.
   */
  public synchronized void update() {
    // calculate dates only once instead of for each node
    mNodeToday = Date.getCurrentDate();
    mNodeTomorrow = mNodeToday.addDays(1);
    mNodeYesterday = mNodeToday.addDays(-1);
    if (mGroupingByDate) {
      createDateNodes();
    }
    else {
      createDefaultNodes();
    }
    UIThreadRunner.invokeLater(new Runnable() {
      @Override
      public void run() {
        PluginTreeModel.getInstance().reload(mDefaultNode);
      }
    });
  }

  /**
   * Add several program as children to this tree node.
   * If the programs are already contained in this sub tree, it is not added again.
   * <br>
   * After you have finished adding all programs, you need to call {@link #update()} to refresh the UI.
   *
   * @param listNew list of new programs
   */
  public synchronized void addPrograms(final List<Program> listNew) {
    Iterator<Program> newIt = listNew.iterator();
    // create sorted lists of current and new programs, but only if this node contains any children at all!
    if (mChildNodes != null && mChildNodes.size() > 0) {
      Program[] currentProgs = getPrograms();
      ArrayList<Program> listCurrent = new ArrayList<Program>(currentProgs.length);
      for (Program currentProg : currentProgs) {
        listCurrent.add(currentProg);
      }
      Comparator<Program> comp = ProgramUtilities.getProgramComparator();
      Collections.sort(listCurrent, comp);
      Collections.sort(listNew, comp);
      Iterator<Program> currentIt = listCurrent.iterator();

      // iterate both lists in parallel and add only new programs
      if (currentIt.hasNext() && newIt.hasNext()) {
        Program newProg = newIt.next();
        Program currentProg = currentIt.next();
        while (newProg != null && currentProg != null) {
          int comparison = comp.compare(newProg, currentProg);
          // new program is sorted first -> add it and investigate next new
          if (comparison < 0) {
            markAndAdd(newProg);
            if (newIt.hasNext()) {
              newProg = newIt.next();
            }
            else {
              newProg = null;
            }
          }
          // old program is sorted first -> go to next old program for comparison
          else if (comparison > 0) {
            if (currentIt.hasNext()) {
              currentProg = currentIt.next();
            }
            else {
              currentProg = null;
            }
          }
          // program already available -> skip
          else if (comparison == 0) {
            if (currentIt.hasNext()) {
              currentProg = currentIt.next();
            }
            else {
              currentProg = null;
            }
            if (newIt.hasNext()) {
              newProg = newIt.next();
            }
            else {
              newProg = null;
            }
          }
        }
      }
    }
    // add all remaining new programs
    while (newIt.hasNext()) {
      markAndAdd(newIt.next());
    }
  }

  private void markAndAdd(final Program program) {
    if (mMarker != null) {
      program.mark(mMarker);
    }
    PluginTreeNode node = new PluginTreeNode(new ProgramItem(program));
    add(node);
  }

  /**
   * Add a single program as child to this tree node.
   * If the program is already contained in this sub tree, it is not added again.
   * <br>
   * After you have finished adding all programs, you need to call {@link #update()} to refresh the UI.
   *
   * @param program
   * @return the tree node containing the program
   */
  public synchronized PluginTreeNode addProgram(final Program program) {
    if (program == null) {
      return null;
    }
    // don't search using contains(), this would require a second search
    PluginTreeNode node = findProgramTreeNode(program, false);
    if (node != null) {
      // node already exists
      return node;
    }

    if (mMarker != null) {
      program.mark(mMarker);
    }
    node = new PluginTreeNode(new ProgramItem(program));
    add(node);
    return node;
  }

  /**
   * Add a single program node as child to this tree node.
   * It is not checked if the program is already contained in this sub tree,
   * so you may only use this method with newly created tree nodes!
   * <br>
   * After you have finished adding all programs, you need to call {@link #update()} to refresh the UI.
   *
   * @param program
   * @return the tree node containing the program
   */
  public synchronized PluginTreeNode addProgramWithoutCheck(
      final Program program) {
    if (mMarker != null) {
      program.mark(mMarker);
    }
    PluginTreeNode node = new PluginTreeNode(new ProgramItem(program));
    add(node);
    return node;
  }

  private PluginTreeNode findProgramTreeNode(final PluginTreeNode root,
      final Program prog, final boolean recursive) {
    if (root.mChildNodes != null) {
      Iterator<PluginTreeNode> it = root.mChildNodes.iterator();
      while (it.hasNext()) {
        PluginTreeNode node = it.next();
        if (!node.isLeaf()) {
          if (recursive) {
            PluginTreeNode n = findProgramTreeNode(node, prog, recursive);
            if (n!=null) {
              return n;
            }
          }
        }
        else {
          ProgramItem item = (ProgramItem)node.getUserObject();
          if (item != null && prog.equals(item.getProgram())) {
            return node;
          }
        }
      }
    }
    return null;
  }

  private PluginTreeNode findProgramTreeNode(final Program prog,
      final boolean recursive) {
    return findProgramTreeNode(this, prog, recursive);
  }


  public synchronized void removeProgram(final ProgramItem item) {
    removeProgram(item.getProgram());
  }

  public synchronized void removeProgram(final Program program) {
    PluginTreeNode node = findProgramTreeNode(program, false);
    if (node != null) {
      mChildNodes.remove(node);
      if (mMarker != null) {
        program.unmark(mMarker);
      }
    }
  }

  public synchronized PluginTreeNode addNode(final String title) {
    PluginTreeNode node = new PluginTreeNode(title);
    add(node);
    return node;
  }

  public ProgramItem[] getProgramItems() {
    // return if there are no child nodes
    if (mChildNodes == null) {
      return new ProgramItem[0];
    }

    // we have child nodes
    ArrayList<Object> list = new ArrayList<Object>();
    addProgramItemsTo(list);

    ProgramItem[] result = new ProgramItem[list.size()];
    list.toArray(result);
    return result;
  }

  private void addProgramItemsTo(final ArrayList<Object> list) {
    if (mChildNodes == null || mChildNodes.isEmpty()) {
      return;
    }
    Iterator<PluginTreeNode> it = mChildNodes.iterator();
    while (it.hasNext()) {
      PluginTreeNode n = it.next();
      if (n.isLeaf()) {
        list.add(n.getUserObject());
      }
      else {
        n.addProgramItemsTo(list);
      }
    }
  }

  public Program[] getPrograms() {
    // return if there are no children
    if (mChildNodes == null) {
      return new Program[0];
    }

    // we have child nodes
    ArrayList<Program> list = new ArrayList<Program>();
    addProgramsTo(list);

    Program[] result = new Program[list.size()];
    list.toArray(result);
    return result;
  }

  private void addProgramsTo(final ArrayList<Program> list) {
    if (mChildNodes == null || mChildNodes.isEmpty()) {
      return;
    }
    Iterator<PluginTreeNode> it = mChildNodes.iterator();
    while (it.hasNext()) {
      PluginTreeNode n = it.next();
      if (n.isLeaf()) {
        final ProgramItem item = (ProgramItem)n.getUserObject();
        final Program program = item.getProgram();
        if (program != null) {
          list.add(program);
        }
      }
      else {
        n.addProgramsTo(list);
      }
    }
  }


  public void store(final ObjectOutputStream out) throws IOException {
    int childrenCnt = 0;
    if (mChildNodes != null) {
      childrenCnt = mChildNodes.size();
    }
    out.writeInt(childrenCnt);

    for (int i=0; i<childrenCnt; i++) {
      PluginTreeNode n = mChildNodes.get(i);
      out.writeInt(n.mNodeType);
      if (n.mNodeType == Node.PROGRAM) {
        ProgramItem item = (ProgramItem) n.getUserObject();
        item.write(out);
      } else {
        String title = (String)n.getUserObject();
        out.writeObject(title);
      }
      n.store(out);
    }
  }

  public void load(final ObjectInputStream in) throws IOException {
    int cnt = in.readInt();
    for (int i=0; i<cnt; i++) {
      int type = in.readInt();
      PluginTreeNode n;
      if (type == Node.PROGRAM) {
        ProgramItem item = new ProgramItem();
        try {
          item.read(in);
        } catch (ClassNotFoundException e) {
          e.printStackTrace();
        }
        Program program = item.getProgram();

        n = new PluginTreeNode(item);
        // only show nodes for programs which still exist
        if (program != null) {
          add(n);
          if (mMarker != null) {
            item.getProgram().mark(mMarker);
          }
        }
      }
      else {
        try {
          String title = (String) in.readObject();
          n = new PluginTreeNode(title);
          add(n);
        } catch (ClassNotFoundException e) {
          e.printStackTrace();
          return;
        }
      }
      n.load(in);
    }
  }


  /**
   * Get the number of child nodes.
   *
   * @return number of child nodes
   */
  public int size() {
    if (mChildNodes == null) {
      return 0;
    }
    return mChildNodes.size();
  }

  public synchronized void clear() {
    if (mChildNodes == null) {
      return;
    }
    mChildNodes.clear();
    mChildNodes = null;
  }

  public boolean isEmpty() {
    return (mChildNodes == null || mChildNodes.isEmpty());
  }

  public boolean isLeaf() {
    return (mDefaultNode.getType() == Node.PROGRAM);
  }

  @Override
  public String toString() {
    switch (mNodeType) {
    case Node.PLUGIN_ROOT: {
      return "plugin node: " + mObject.toString();
    }
    case Node.PROGRAM: {
      return "program node: " + mObject.toString();
    }
    case Node.STRUCTURE_NODE: {
      return "structure node: " + mObject.toString();
    }
    case Node.CUSTOM_NODE: {
      return "custom node: " + mObject.toString();
    }
    }
    return super.toString();
  }

  public static class RemovedProgramsHandler {
    private ArrayList<Program> mProgArr;
    public RemovedProgramsHandler() {
      mProgArr = new ArrayList<Program>();
    }
    public void clear() {
      mProgArr.clear();
    }

    public void addRemovedProgram(final Program prog) {
      mProgArr.add(prog);
    }

    public Program[] getRemovedPrograms() {
      Program[] progArr = new Program[mProgArr.size()];
      mProgArr.toArray(progArr);
      return progArr;
    }
  }

  /**
   * @return copy of child node collection
   * @since 3.0
   */
  public PluginTreeNode[] getChildren() {
    if (mChildNodes == null) {
      return new PluginTreeNode[0];
    }
    return mChildNodes.toArray(new PluginTreeNode[mChildNodes.size()]);
  }

  @Override
  public int compareTo(final PluginTreeNode other) {
    final Object otherUserObject = other.getUserObject();
    if (mObject instanceof ProgramItem
        && otherUserObject instanceof ProgramItem) {
      return ((ProgramItem) mObject).compareTo((ProgramItem) otherUserObject);
    }
    if (mObject instanceof String && otherUserObject instanceof String) {
      return ((String) mObject).compareToIgnoreCase((String) otherUserObject);
    }
    if (mObject instanceof String) {
      return 1;
    }
    return -1;
  }
}
TOP

Related Classes of devplugin.PluginTreeNode$RemovedProgramsHandler

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.