/******************************************************************************* * Copyright (c) 2013, 2014, 2015 Dirk Fauth and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Dirk Fauth <dirk.fauth@googlemail.com> - initial API and implementation * Dirk Fauth <dirk.fauth@googlemail.com> - modified example for correct unique handling of elements *******************************************************************************/ package org.eclipse.nebula.widgets.nattable.examples._600_GlazedLists._604_Tree; import java.util.Comparator; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.concurrent.atomic.AtomicInteger; import org.eclipse.jface.layout.GridDataFactory; import org.eclipse.nebula.widgets.nattable.NatTable; import org.eclipse.nebula.widgets.nattable.config.AbstractRegistryConfiguration; import org.eclipse.nebula.widgets.nattable.config.AbstractUiBindingConfiguration; import org.eclipse.nebula.widgets.nattable.config.CellConfigAttributes; import org.eclipse.nebula.widgets.nattable.config.ConfigRegistry; import org.eclipse.nebula.widgets.nattable.config.DefaultNatTableStyleConfiguration; import org.eclipse.nebula.widgets.nattable.config.IConfigRegistry; import org.eclipse.nebula.widgets.nattable.data.IColumnPropertyAccessor; import org.eclipse.nebula.widgets.nattable.data.IDataProvider; import org.eclipse.nebula.widgets.nattable.data.IRowDataProvider; import org.eclipse.nebula.widgets.nattable.data.ListDataProvider; import org.eclipse.nebula.widgets.nattable.data.ReflectiveColumnPropertyAccessor; import org.eclipse.nebula.widgets.nattable.data.convert.DefaultDateDisplayConverter; import org.eclipse.nebula.widgets.nattable.dataset.person.PersonService; import org.eclipse.nebula.widgets.nattable.dataset.person.PersonWithAddress; import org.eclipse.nebula.widgets.nattable.examples.AbstractNatExample; import org.eclipse.nebula.widgets.nattable.examples.runner.StandaloneNatExampleRunner; import org.eclipse.nebula.widgets.nattable.extension.glazedlists.GlazedListsEventLayer; import org.eclipse.nebula.widgets.nattable.extension.glazedlists.tree.GlazedListTreeData; import org.eclipse.nebula.widgets.nattable.extension.glazedlists.tree.GlazedListTreeRowModel; import org.eclipse.nebula.widgets.nattable.grid.data.DefaultColumnHeaderDataProvider; import org.eclipse.nebula.widgets.nattable.grid.data.DefaultCornerDataProvider; import org.eclipse.nebula.widgets.nattable.grid.data.DefaultRowHeaderDataProvider; import org.eclipse.nebula.widgets.nattable.grid.layer.ColumnHeaderLayer; import org.eclipse.nebula.widgets.nattable.grid.layer.CornerLayer; import org.eclipse.nebula.widgets.nattable.grid.layer.DefaultColumnHeaderDataLayer; import org.eclipse.nebula.widgets.nattable.grid.layer.DefaultRowHeaderDataLayer; import org.eclipse.nebula.widgets.nattable.grid.layer.GridLayer; import org.eclipse.nebula.widgets.nattable.grid.layer.RowHeaderLayer; import org.eclipse.nebula.widgets.nattable.layer.AbstractLayerTransform; import org.eclipse.nebula.widgets.nattable.layer.DataLayer; import org.eclipse.nebula.widgets.nattable.layer.ILayer; import org.eclipse.nebula.widgets.nattable.layer.LabelStack; import org.eclipse.nebula.widgets.nattable.layer.cell.AbstractOverrider; import org.eclipse.nebula.widgets.nattable.layer.cell.ILayerCell; import org.eclipse.nebula.widgets.nattable.painter.cell.BackgroundPainter; import org.eclipse.nebula.widgets.nattable.painter.cell.CheckBoxPainter; import org.eclipse.nebula.widgets.nattable.painter.cell.ICellPainter; import org.eclipse.nebula.widgets.nattable.painter.cell.decorator.PaddingDecorator; import org.eclipse.nebula.widgets.nattable.selection.SelectionLayer; import org.eclipse.nebula.widgets.nattable.style.DisplayMode; import org.eclipse.nebula.widgets.nattable.tree.ITreeRowModel; import org.eclipse.nebula.widgets.nattable.tree.TreeLayer; import org.eclipse.nebula.widgets.nattable.tree.command.TreeCollapseAllCommand; import org.eclipse.nebula.widgets.nattable.tree.command.TreeExpandAllCommand; import org.eclipse.nebula.widgets.nattable.tree.command.TreeExpandToLevelCommand; import org.eclipse.nebula.widgets.nattable.tree.config.TreeConfigAttributes; import org.eclipse.nebula.widgets.nattable.tree.painter.IndentedTreeImagePainter; import org.eclipse.nebula.widgets.nattable.tree.painter.TreeImagePainter; import org.eclipse.nebula.widgets.nattable.ui.NatEventData; import org.eclipse.nebula.widgets.nattable.ui.binding.UiBindingRegistry; import org.eclipse.nebula.widgets.nattable.ui.matcher.MouseEventMatcher; import org.eclipse.nebula.widgets.nattable.ui.menu.IMenuItemProvider; import org.eclipse.nebula.widgets.nattable.ui.menu.IMenuItemState; import org.eclipse.nebula.widgets.nattable.ui.menu.MenuItemProviders; import org.eclipse.nebula.widgets.nattable.ui.menu.PopupMenuAction; import org.eclipse.nebula.widgets.nattable.ui.menu.PopupMenuBuilder; import org.eclipse.nebula.widgets.nattable.ui.util.CellEdgeEnum; import org.eclipse.nebula.widgets.nattable.util.GUIHelper; import org.eclipse.nebula.widgets.nattable.viewport.ViewportLayer; import org.eclipse.swt.SWT; import org.eclipse.swt.events.SelectionAdapter; import org.eclipse.swt.events.SelectionEvent; import org.eclipse.swt.layout.GridLayout; import org.eclipse.swt.layout.RowLayout; import org.eclipse.swt.widgets.Button; import org.eclipse.swt.widgets.Composite; import org.eclipse.swt.widgets.Control; import org.eclipse.swt.widgets.Menu; import org.eclipse.swt.widgets.MenuItem; import ca.odell.glazedlists.EventList; import ca.odell.glazedlists.GlazedLists; import ca.odell.glazedlists.SortedList; import ca.odell.glazedlists.TransformedList; import ca.odell.glazedlists.TreeList; /** * Simple example showing how to create a tree within a grid. */ public class _6042_TreeStructureGridExample extends AbstractNatExample { public static final String MARRIED_LABEL = "marriedLabel"; public static final String DATE_LABEL = "dateLabel"; public static void main(String[] args) throws Exception { StandaloneNatExampleRunner.run(new _6042_TreeStructureGridExample()); } @Override public String getDescription() { return "This example shows how to create a tree within a grid." + " It creates a tree structure where the tree nodes are newly" + " added elements that contain all children. It also shows" + " how to create a multi level tree and how to exchange the" + " tree painter via configuration."; } @Override public Control createExampleControl(Composite parent) { Composite container = new Composite(parent, SWT.NONE); container.setLayout(new GridLayout()); // create a new ConfigRegistry which will be needed for GlazedLists // handling ConfigRegistry configRegistry = new ConfigRegistry(); // property names of the Person class String[] propertyNames = { "lastName", "firstName", "gender", "married", "birthday" }; // mapping from property to label, needed for column header labels Map<String, String> propertyToLabelMap = new HashMap<>(); propertyToLabelMap.put("lastName", "Lastname"); propertyToLabelMap.put("firstName", "Firstname"); propertyToLabelMap.put("gender", "Gender"); propertyToLabelMap.put("married", "Married"); propertyToLabelMap.put("birthday", "Birthday"); IColumnPropertyAccessor<PersonWithAddress> columnPropertyAccessor = new ReflectiveColumnPropertyAccessor<>(propertyNames); final BodyLayerStack bodyLayerStack = new BodyLayerStack( PersonService.getPersonsWithAddress(5), columnPropertyAccessor, new PersonWithAddressTwoLevelTreeFormat()); // new PersonWithAddressTreeFormat()); // build the column header layer IDataProvider columnHeaderDataProvider = new DefaultColumnHeaderDataProvider(propertyNames, propertyToLabelMap); DataLayer columnHeaderDataLayer = new DefaultColumnHeaderDataLayer(columnHeaderDataProvider); ILayer columnHeaderLayer = new ColumnHeaderLayer(columnHeaderDataLayer, bodyLayerStack, bodyLayerStack.getSelectionLayer()); // build the row header layer IDataProvider rowHeaderDataProvider = new DefaultRowHeaderDataProvider(bodyLayerStack.getBodyDataProvider()); DataLayer rowHeaderDataLayer = new DefaultRowHeaderDataLayer(rowHeaderDataProvider); ILayer rowHeaderLayer = new RowHeaderLayer(rowHeaderDataLayer, bodyLayerStack, bodyLayerStack.getSelectionLayer()); // build the corner layer IDataProvider cornerDataProvider = new DefaultCornerDataProvider(columnHeaderDataProvider, rowHeaderDataProvider); DataLayer cornerDataLayer = new DataLayer(cornerDataProvider); ILayer cornerLayer = new CornerLayer(cornerDataLayer, rowHeaderLayer, columnHeaderLayer); // build the grid layer GridLayer gridLayer = new GridLayer(bodyLayerStack, columnHeaderLayer, rowHeaderLayer, cornerLayer); // turn the auto configuration off as we want to add our header menu // configuration final NatTable natTable = new NatTable(container, gridLayer, false); // as the autoconfiguration of the NatTable is turned off, we have to // add the DefaultNatTableStyleConfiguration and the ConfigRegistry // manually natTable.setConfigRegistry(configRegistry); natTable.addConfiguration(new DefaultNatTableStyleConfiguration()); natTable.addConfiguration(new AbstractRegistryConfiguration() { @Override public void configureRegistry(IConfigRegistry configRegistry) { // register a CheckBoxPainter as CellPainter for the married // information configRegistry.registerConfigAttribute( CellConfigAttributes.CELL_PAINTER, new CheckBoxPainter(), DisplayMode.NORMAL, MARRIED_LABEL); configRegistry.registerConfigAttribute( CellConfigAttributes.DISPLAY_CONVERTER, new DefaultDateDisplayConverter("MM/dd/yyyy"), DisplayMode.NORMAL, DATE_LABEL); // exchange the painter that is used to render the tree // structure the following will use triangles instead of // plus/minus icons to show the tree structure and // expand/collapse state and adds padding between cell // border and tree icons. TreeImagePainter treeImagePainter = new TreeImagePainter( false, GUIHelper.getImage("right"), //$NON-NLS-1$ GUIHelper.getImage("right_down"), null); //$NON-NLS-1$ ICellPainter treeStructurePainter = new BackgroundPainter( new PaddingDecorator( new IndentedTreeImagePainter(10, null, CellEdgeEnum.LEFT, treeImagePainter, false, 2, true), 0, 5, 0, 5, false)); configRegistry.registerConfigAttribute( TreeConfigAttributes.TREE_STRUCTURE_PAINTER, treeStructurePainter, DisplayMode.NORMAL); } }); natTable.addConfiguration(new TreeDebugMenuConfiguration(natTable)); natTable.configure(); GridDataFactory.fillDefaults().grab(true, true).applyTo(natTable); Composite buttonPanel = new Composite(container, SWT.NONE); buttonPanel.setLayout(new RowLayout()); GridDataFactory.fillDefaults().grab(true, false).applyTo(buttonPanel); Button collapseAllButton = new Button(buttonPanel, SWT.PUSH); collapseAllButton.setText("Collapse All"); collapseAllButton.addSelectionListener(new SelectionAdapter() { @Override public void widgetSelected(SelectionEvent e) { natTable.doCommand(new TreeCollapseAllCommand()); } }); Button expandAllButton = new Button(buttonPanel, SWT.PUSH); expandAllButton.setText("Expand All"); expandAllButton.addSelectionListener(new SelectionAdapter() { @Override public void widgetSelected(SelectionEvent e) { natTable.doCommand(new TreeExpandAllCommand()); } }); Button expandToLevelButton = new Button(buttonPanel, SWT.PUSH); expandToLevelButton.setText("Expand To Level"); expandToLevelButton.addSelectionListener(new SelectionAdapter() { @Override public void widgetSelected(SelectionEvent e) { natTable.doCommand(new TreeExpandToLevelCommand(1)); } }); return container; } /** * Always encapsulate the body layer stack in an AbstractLayerTransform to * ensure that the index transformations are performed in later commands. * * @param <PersonWithAddress> */ @SuppressWarnings({ "unchecked", "rawtypes" }) class BodyLayerStack extends AbstractLayerTransform { private final TreeList treeList; private final IRowDataProvider bodyDataProvider; private final SelectionLayer selectionLayer; public BodyLayerStack( List<PersonWithAddress> values, IColumnPropertyAccessor<PersonWithAddress> columnPropertyAccessor, TreeList.Format treeFormat) { // wrapping of the list to show into GlazedLists // see http://publicobject.com/glazedlists/ for further information EventList<PersonWithAddress> eventList = GlazedLists.eventList(values); TransformedList<PersonWithAddress, PersonWithAddress> rowObjectsGlazedList = GlazedLists.threadSafeList(eventList); // use the SortedList constructor with 'null' for the Comparator // because the Comparator // will be set by configuration SortedList<PersonWithAddress> sortedList = new SortedList<>(rowObjectsGlazedList, null); // wrap the SortedList with the TreeList this.treeList = new TreeList(sortedList, treeFormat, TreeList.nodesStartExpanded()); this.bodyDataProvider = new ListDataProvider<Object>( this.treeList, new PersonWithAddressTreeColumnPropertyAccessor(columnPropertyAccessor)); DataLayer bodyDataLayer = new DataLayer(this.bodyDataProvider); // simply apply labels for every column by index bodyDataLayer.setConfigLabelAccumulator(new AbstractOverrider() { @Override public void accumulateConfigLabels(LabelStack configLabels, int columnPosition, int rowPosition) { Object rowObject = BodyLayerStack.this.bodyDataProvider.getRowObject(rowPosition); if (rowObject instanceof PersonWithAddress) { if (columnPosition == 3) { configLabels.addLabel(MARRIED_LABEL); } else if (columnPosition == 4) { configLabels.addLabel(DATE_LABEL); } } } }); // layer for event handling of GlazedLists and PropertyChanges GlazedListsEventLayer<PersonWithAddress> glazedListsEventLayer = new GlazedListsEventLayer<PersonWithAddress>(bodyDataLayer, this.treeList); GlazedListTreeData<Object> treeData = new GlazedListTreeData<Object>(this.treeList); ITreeRowModel<Object> treeRowModel = new GlazedListTreeRowModel<>(treeData); // ITreeRowModel<Object> treeRowModel = new // TreeRowModel<Object>(treeData); this.selectionLayer = new SelectionLayer(glazedListsEventLayer); TreeLayer treeLayer = new TreeLayer(this.selectionLayer, treeRowModel); ViewportLayer viewportLayer = new ViewportLayer(treeLayer); setUnderlyingLayer(viewportLayer); } public SelectionLayer getSelectionLayer() { return this.selectionLayer; } public TreeList<PersonWithAddress> getTreeList() { return this.treeList; } public IDataProvider getBodyDataProvider() { return this.bodyDataProvider; } } private class PersonWithAddressTreeColumnPropertyAccessor implements IColumnPropertyAccessor<Object> { private IColumnPropertyAccessor<PersonWithAddress> cpa; public PersonWithAddressTreeColumnPropertyAccessor(IColumnPropertyAccessor<PersonWithAddress> cpa) { this.cpa = cpa; } @Override public Object getDataValue(Object rowObject, int columnIndex) { if (rowObject instanceof PersonWithAddress) { return this.cpa.getDataValue((PersonWithAddress) rowObject, columnIndex); } else if (columnIndex == 0) { return rowObject; } return null; } @Override public void setDataValue(Object rowObject, int columnIndex, Object newValue) { if (rowObject instanceof PersonWithAddress) { this.cpa.setDataValue((PersonWithAddress) rowObject, columnIndex, newValue); } } @Override public int getColumnCount() { return this.cpa.getColumnCount(); } @Override public String getColumnProperty(int columnIndex) { return this.cpa.getColumnProperty(columnIndex); } @Override public int getColumnIndex(String propertyName) { return this.cpa.getColumnIndex(propertyName); } } @SuppressWarnings("unused") private class PersonWithAddressTreeFormat implements TreeList.Format<Object> { @Override public void getPath(List<Object> path, Object element) { if (element instanceof PersonWithAddress) { PersonWithAddress ele = (PersonWithAddress) element; path.add(new LastNameGroup(ele.getId(), ele.getLastName())); } path.add(element); } @Override public boolean allowsChildren(Object element) { return true; } @Override public Comparator<? super Object> getComparator(int depth) { return new Comparator<Object>() { @Override public int compare(Object o1, Object o2) { String e1 = (o1 instanceof PersonWithAddress) ? ((PersonWithAddress) o1).getLastName() : o1.toString(); String e2 = (o2 instanceof PersonWithAddress) ? ((PersonWithAddress) o2).getLastName() : o2.toString(); return e1.compareTo(e2); } }; } } private class PersonWithAddressTwoLevelTreeFormat implements TreeList.Format<Object> { AtomicInteger counter = new AtomicInteger(); Map<String, LastNameGroup> lastNames = new HashMap<>(); Map<String, FirstNameGroup> firstNames = new HashMap<>(); @Override public void getPath(List<Object> path, Object element) { if (element instanceof PersonWithAddress) { PersonWithAddress ele = (PersonWithAddress) element; if (!this.lastNames.containsKey(ele.getLastName())) { this.lastNames.put(ele.getLastName(), new LastNameGroup(this.counter.incrementAndGet(), ele.getLastName())); } path.add(this.lastNames.get(ele.getLastName())); String firstNameKey = ele.getLastName() + "_" + ele.getFirstName(); if (!this.firstNames.containsKey(firstNameKey)) { this.firstNames.put(firstNameKey, new FirstNameGroup(ele.getLastName(), ele.getFirstName())); } path.add(this.firstNames.get(firstNameKey)); } path.add(element); } @Override public boolean allowsChildren(Object element) { return true; } @Override public Comparator<? super Object> getComparator(final int depth) { return new Comparator<Object>() { @Override public int compare(Object o1, Object o2) { String e1 = (o1 instanceof PersonWithAddress) ? (depth == 0 ? ((PersonWithAddress) o1).getLastName() : ((PersonWithAddress) o1).getFirstName()) : o1.toString(); String e2 = (o2 instanceof PersonWithAddress) ? (depth == 0 ? ((PersonWithAddress) o2).getLastName() : ((PersonWithAddress) o2).getFirstName()) : o2.toString(); return e1.compareTo(e2); } }; } } // To make expand/collapse work correctly with the TreeRowModel, the // elements in the TreeList needs to be identifiable. This is necessary so // List#indexOf() returns the correct positions class LastNameGroup implements Comparable<LastNameGroup> { int id; String lastName; public LastNameGroup(int id, String lastName) { this.id = id; this.lastName = lastName; } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + this.id; result = prime * result + ((this.lastName == null) ? 0 : this.lastName.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; LastNameGroup other = (LastNameGroup) obj; if (this.id != other.id) return false; if (this.lastName == null) { if (other.lastName != null) return false; } else if (!this.lastName.equals(other.lastName)) return false; return true; } @Override public String toString() { return this.lastName; } @Override public int compareTo(LastNameGroup o) { return this.lastName.compareTo(o.lastName); } } // firstname group is unique within a lastname group class FirstNameGroup implements Comparable<FirstNameGroup> { String lastName; String firstName; public FirstNameGroup(String lastName, String firstName) { this.lastName = lastName; this.firstName = firstName; } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((this.firstName == null) ? 0 : this.firstName.hashCode()); result = prime * result + ((this.lastName == null) ? 0 : this.lastName.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; FirstNameGroup other = (FirstNameGroup) obj; if (this.firstName == null) { if (other.firstName != null) return false; } else if (!this.firstName.equals(other.firstName)) return false; if (this.lastName == null) { if (other.lastName != null) return false; } else if (!this.lastName.equals(other.lastName)) return false; return true; } @Override public String toString() { return this.firstName; } @Override public int compareTo(FirstNameGroup o) { return this.firstName.compareTo(o.firstName); } } class TreeDebugMenuConfiguration extends AbstractUiBindingConfiguration { private final Menu menu; public TreeDebugMenuConfiguration(final NatTable natTable) { this.menu = new PopupMenuBuilder(natTable) .withMenuItemProvider("expandToLevel", new IMenuItemProvider() { @Override public void addMenuItem(final NatTable natTable, Menu popupMenu) { MenuItem menuItem = new MenuItem(popupMenu, SWT.PUSH); menuItem.setText("Expand 1 level"); menuItem.setEnabled(true); menuItem.addSelectionListener(new SelectionAdapter() { @Override public void widgetSelected(SelectionEvent event) { NatEventData eventData = MenuItemProviders.getNatEventData(event); int rowIndex = natTable.getRowIndexByPosition(eventData.getRowPosition()); natTable.doCommand(new TreeExpandToLevelCommand(rowIndex, 1)); } }); } }) .withVisibleState("expandToLevel", new IMenuItemState() { @Override public boolean isActive(NatEventData natEventData) { ILayerCell cell = natTable.getCellByPosition( natEventData.getColumnPosition(), natEventData.getRowPosition()); return cell.getConfigLabels().hasLabel(TreeLayer.TREE_COLUMN_CELL); } }) .withInspectLabelsMenuItem() .build(); } @Override public void configureUiBindings(UiBindingRegistry uiBindingRegistry) { uiBindingRegistry.registerMouseDownBinding( new MouseEventMatcher(SWT.NONE, null, 3), new PopupMenuAction(this.menu)); } } }