/** * OrbisGIS is a java GIS application dedicated to research in GIScience. * OrbisGIS is developed by the GIS group of the DECIDE team of the * Lab-STICC CNRS laboratory, see <http://www.lab-sticc.fr/>. * * The GIS group of the DECIDE team is located at : * * Laboratoire Lab-STICC – CNRS UMR 6285 * Equipe DECIDE * UNIVERSITÉ DE BRETAGNE-SUD * Institut Universitaire de Technologie de Vannes * 8, Rue Montaigne - BP 561 56017 Vannes Cedex * * OrbisGIS is distributed under GPL 3 license. * * Copyright (C) 2007-2014 CNRS (IRSTV FR CNRS 2488) * Copyright (C) 2015-2017 CNRS (Lab-STICC UMR CNRS 6285) * * This file is part of OrbisGIS. * * OrbisGIS 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 3 of the License, or (at your option) any later * version. * * OrbisGIS 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 * OrbisGIS. If not, see <http://www.gnu.org/licenses/>. * * For more information, please consult: <http://www.orbisgis.org/> * or contact directly: * info_at_ orbisgis.org */ package org.orbisgis.view.toc.actions.cui; import java.awt.*; import java.beans.EventHandler; import java.beans.PropertyChangeEvent; import java.beans.PropertyChangeListener; import java.net.URL; import java.sql.Connection; import java.sql.SQLException; import java.util.LinkedList; import java.util.List; import javax.swing.JLabel; import javax.swing.JPanel; import javax.swing.JScrollPane; import javax.swing.JSplitPane; import javax.swing.SwingConstants; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.orbisgis.corejdbc.MetaData; import org.orbisgis.coremap.layerModel.ILayer; import org.orbisgis.coremap.map.MapTransform; import org.orbisgis.coremap.renderer.se.Rule; import org.orbisgis.coremap.renderer.se.Style; import org.orbisgis.coremap.renderer.se.Symbolizer; import org.orbisgis.legend.Legend; import org.orbisgis.legend.thematic.factory.LegendFactory; import org.orbisgis.legend.thematic.recode.AbstractRecodedLegend; import org.orbisgis.sif.UIFactory; import org.orbisgis.sif.UIPanel; import org.orbisgis.view.toc.actions.cui.legend.ILegendPanel; import org.orbisgis.view.toc.actions.cui.legend.ILegendPanelFactory; import org.orbisgis.view.toc.actions.cui.legend.ISELegendPanel; import org.orbisgis.view.toc.actions.cui.legend.ui.PnlRule; import org.orbisgis.view.toc.actions.cui.legend.ui.PnlStyle; import org.orbisgis.view.toc.wrapper.RuleWrapper; import org.orbisgis.view.toc.wrapper.StyleWrapper; import org.xnap.commons.i18n.I18n; import org.xnap.commons.i18n.I18nFactory; /** * The Simple Style Editor user interface. * * @author Alexis Guéganno * @author Adam Gouge * @author others */ public class SimpleStyleEditor extends JPanel implements UIPanel, LegendContext { /** * Tree to display legends. */ private LegendTree legendTree; // ********** GRAPHICS ************** /** * Used to switch between legend instances. */ private CardLayout cardLayout; /** * Dialog panel. */ private JPanel dialogContainer; // ********** INITIALIZATION VARIABLES ************** /** * Represents the current state of the map. */ private MapTransform mt; /** * The type of the Geometry of the legend's layer currently being * modified. {@link org.h2gis.utilities.GeometryTypeCodes} */ private int type; /** * The layer we are editing. */ private ILayer layer; // ********** INITIALIZATION-LIKE VARIABLES ********** /** * {@link SimpleGeometryType}. */ private int geometryType; /** * {@link StyleWrapper} for the {@link Style}s of the layer we are editing. */ private StyleWrapper styleWrapper; // ********** IDS ************** /** * Id for the {@link CardLayout} panel to be shown when there is no legend. */ private static final String NO_LEGEND_ID = "no-legend"; /** * Id of the most recently added {@link CardLayout} panel. */ private static String lastUID = ""; // ********** OTHER ************** /** * Translator. */ private static final I18n I18N = I18nFactory. getI18n(SimpleStyleEditor.class); /** * Scroll increment (To fix slow scrolling). */ private static final int INCREMENT = 16; private static final Logger LOGGER = LoggerFactory.getLogger(SimpleStyleEditor.class); /** * Constructor * * @param mt Map transform * @param type Layer geometry type * @param layer Layer * @param style Style */ public SimpleStyleEditor(MapTransform mt, int type, ILayer layer, Style style) { // Set the layout. super(new BorderLayout()); // Recover the first three paramters. this.mt = mt; this.type = type; this.layer = layer; // Get the geometry type and available legends. this.geometryType = SimpleGeometryType.getSimpleType(type); // Initialize the dialog container, adding the empty dialog. cardLayout = new CardLayout(); dialogContainer = new JPanel(cardLayout); dialogContainer.setPreferredSize(new Dimension(640, 420)); addEmptyDialog(); // Add all panels. styleWrapper = addAllPanels(style); // Initialize the legend tree. legendTree = new LegendTree(this); // Put everything inside a split pane. JSplitPane splitPane = new JSplitPane(JSplitPane.HORIZONTAL_SPLIT, legendTree, dialogContainer); // Should be just wide enough for "Interval classification - Point" splitPane.setDividerLocation(260); Dimension minimumSize = new Dimension(100, 50); legendTree.setMinimumSize(minimumSize); dialogContainer.setMinimumSize(minimumSize); add(splitPane, BorderLayout.CENTER); } /** * Adds the empty dialog to the card layout. */ private void addEmptyDialog() { JPanel textHolder = new JPanel(new BorderLayout()); JLabel text = new JLabel(I18N.tr("Add or select a legend.")); text.setHorizontalAlignment(SwingConstants.CENTER); textHolder.add(text, BorderLayout.CENTER); dialogContainer.add(NO_LEGEND_ID, textHolder); } /** * Adds the style, rule and symbol panels for the given style and returns * the corresponding {@link StyleWrapper}. * * @param style Style * * @return The {@link StyleWrapper} corresponding to the style panel * * @see #addRuleAndSymbolPanels(org.orbisgis.coremap.renderer.se.Style) */ private StyleWrapper addAllPanels(Style style) { // Get a style wrapper based on this style and the previous list // of rule wrappers constructed in addRuleAndSymbolPanels. StyleWrapper sw = new StyleWrapper(style, addRuleAndSymbolPanels(style)); addStylePanel(sw); return sw; } /** * Adds the style panel attached to the given {@link StyleWrapper}. * * @param sw StyleWrapper */ private void addStylePanel(StyleWrapper sw) { PnlStyle stylePanel = sw.getPanel(); stylePanel.setId(createNewID()); stylePanel.addPropertyChangeListener( EventHandler.create(PropertyChangeListener.class, this, "onNodeNameChange", "")); dialogContainer.add(stylePanel.getId(), getJScrollPane(stylePanel)); } /** * Gets a new {@link JScrollPane} on the given panel, setting the scroll * increments appropriately. * * @param panel Panel * * @return A new {@link JScrollPane} on the given panel */ private JScrollPane getJScrollPane(ISELegendPanel panel) { JScrollPane jsp = new JScrollPane(panel.getComponent()); jsp.getHorizontalScrollBar().setUnitIncrement(INCREMENT); jsp.getVerticalScrollBar().setUnitIncrement(INCREMENT); return jsp; } /** * Adds the rule and symbol panels for the rules and symbols attached to the * given style and returns a list of the corresponding {@link RuleWrapper}s. * * @param style Style * * @return A list of {@link RuleWrapper}s corresponding to the newly added * rule panels. * * @see #addSymbolPanels(org.orbisgis.coremap.renderer.se.Rule) */ private List<RuleWrapper> addRuleAndSymbolPanels(Style style) { List<Rule> rules = style.getRules(); List<RuleWrapper> ruleWrapperList = new LinkedList<RuleWrapper>(); for (int i = 0; i < rules.size(); i++) { // Get the rule. Rule rule = rules.get(i); // Get a new RuleWrapper based on this rule and the list of // symbol panels constructed in addSymbolPanels. RuleWrapper ruleWrapper = new RuleWrapper(this, rule, addSymbolPanels(rule)); addRulePanel(ruleWrapper); // Add the rule wrapper panel to the list of rule wrapper panels. ruleWrapperList.add(ruleWrapper); } return ruleWrapperList; } /** * Adds the rule panel attached to the given {@link RuleWrapper}. * * @param ruleWrapper RuleWrapper */ private void addRulePanel(RuleWrapper ruleWrapper) { // Get the panel associated to this RuleWrapper, set its id, // initialize it and add a listener for when its node name changes. PnlRule rulePanel = ruleWrapper.getPanel(); rulePanel.setId(createNewID()); rulePanel.addPropertyChangeListener( EventHandler.create(PropertyChangeListener.class, this, "onNodeNameChange", "")); // Add the rule wrapper panel to the container after putting it in // a new JScrollPane. dialogContainer.add(rulePanel.getId(), getJScrollPane(rulePanel)); } /** * Adds the symbol panels for the legends (=symbols) attached to the given * rule and returns a list of the corresponding {@link ILegendPanel}s. * * @param rule Rule * * @return A list of {@link ILegendPanel}s corresponding to the newly added * symbol panels. */ private List<ILegendPanel> addSymbolPanels(Rule rule) { List<ILegendPanel> symbolPanelList = new LinkedList<ILegendPanel>(); // For each symbol in this rule, add its symbol panel to the list of // symbol panels. for (Symbolizer symb : rule.getCompositeSymbolizer(). getSymbolizerList()) { symbolPanelList.add(addSymbolPanel(symb)); } return symbolPanelList; } /** * Adds the symbol panel attached to the given {@link Symbolizer} and * returns the panel. * * @param symb Symbolizer * * @return The newly generated symbol panel */ private ILegendPanel addSymbolPanel(Symbolizer symb) { // Get the legend corresponding to this symbolizer. Legend legend = LegendFactory.getLegend(symb); if(legend instanceof AbstractRecodedLegend){ String table = layer.getTableReference(); AbstractRecodedLegend leg = (AbstractRecodedLegend) legend; try(Connection connection = layer.getDataManager().getDataSource().getConnection()) { String f = leg.getLookupFieldName(); int type = MetaData.getFieldType(connection, table, f); leg.setComparator(AbstractRecodedLegend.getComparator(type)); } catch (SQLException e) { LOGGER.warn("Can't retrieve an accurate Comparator for this classification"); } } // Initialize a panel for this legend. ILegendPanel panel = ILegendPanelFactory.getILegendPanel(this, legend); // Give it a new id. panel.setId(createNewID()); // Add the symbol panel to the container after putting it in a // new JScrollPane. dialogContainer.add(panel.getId(), getJScrollPane(panel)); return panel; } /** * Creates a new unique ID for retrieving panels in the card layout. * * @return A new unique ID */ public static String createNewID() { String name = "orbisgis" + System.currentTimeMillis(); while (name.equals(lastUID)) { name = "" + System.currentTimeMillis(); } lastUID = name; return name; } /** * Shows the dialog for the given legend in the card layout. */ protected void showDialogForLegend(ISELegendPanel selected) { cardLayout.show(dialogContainer, selected.getId()); } /** * Retrieves the currently selected legend in the tree and shows the * corresponding dialog in the card layout; shows the empty panel if no * legend is selected. */ protected void showDialogForCurrentlySelectedLegend() { ISELegendPanel selected = legendTree.getSelectedPanel(); if (selected != null) { cardLayout.show(dialogContainer, selected.getId()); } else { cardLayout.show(dialogContainer, NO_LEGEND_ID); } } // ******************* EventHandler methods ********************** /** * Updates the name of the selected element when it changes. * * @param pce The original event */ public void onNodeNameChange(PropertyChangeEvent pce) { if (pce.getPropertyName().equals(PnlRule.NAME_PROPERTY)) { legendTree.selectedNameChanged(); } } /** * Initialize the given panel with this as {@link LegendContext}, set its * id, add it to the container inside a {@link JScrollPane} and show its * dialog. * * @param panel The panel to initialize and show */ // TODO: Should this method be moved to LegendTree? public void legendAdded(ISELegendPanel panel) { panel.setId(createNewID()); dialogContainer.add(panel.getId(), getJScrollPane(panel)); showDialogForCurrentlySelectedLegend(); } /** * Remove the given panel from the card layout and refresh the display. * * @param panel Panel */ public void legendRemoved(ISELegendPanel panel) { cardLayout.removeLayoutComponent(panel.getComponent()); showDialogForCurrentlySelectedLegend(); } // TODO: Find usages / Document. public void legendRenamed(int idx, String newName) { showDialogForCurrentlySelectedLegend(); } // TODO: Find usages / Document. public void legendSelected() { showDialogForCurrentlySelectedLegend(); } // ************************* Getters ***************************** /** * Gets the style wrapper. * * @return The style wrapper */ public StyleWrapper getStyleWrapper() { return styleWrapper; } // ****************** LegendContext methods ********************** @Override public int getGeometryType() { return geometryType; } @Override public boolean isPoint() { return (geometryType & SimpleGeometryType.POINT) > 0; } @Override public boolean isLine() { return (geometryType & SimpleGeometryType.LINE) > 0; } @Override public boolean isPolygon() { return (geometryType & SimpleGeometryType.POLYGON) > 0; } @Override public ILayer getLayer() { return layer; } @Override public MapTransform getCurrentMapTransform() { return mt; } // ********************** UIPanel methods ************************ @Override public URL getIconURL() { return UIFactory.getDefaultIcon(); } @Override public String getTitle() { return I18N.tr("Simple Style Editor - {0}", layer.getName()); } @Override public String validateInput() { if (!legendTree.hasLegend()) { return I18N.tr("You must create at least one legend"); } List<String> errors = styleWrapper.validateInput(); StringBuilder sb = new StringBuilder(); for (String message : errors) { if (message != null && !message.isEmpty()) { sb.append(message); sb.append("\n"); } } String err = sb.toString(); if (err != null && !err.isEmpty()) { return err; } return null; } @Override public Component getComponent() { return this; } }