/[schmitzm]/branches/2.0-RC2/src/skrueger/geotools/selection/FeatureMapLayerSelectionSynchronizer.java
ViewVC logotype

Diff of /branches/2.0-RC2/src/skrueger/geotools/selection/FeatureMapLayerSelectionSynchronizer.java

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

trunk/src/skrueger/geotools/selection/FeatureMapLayerSelectionSynchronizer.java revision 162 by alfonx, Mon Jun 22 21:05:20 2009 UTC branches/1.0-gt2-2.6/src/skrueger/geotools/selection/FeatureMapLayerSelectionSynchronizer.java revision 493 by mojays, Fri Oct 23 14:09:27 2009 UTC
# Line 1  Line 1 
1    /*******************************************************************************
2     * Copyright (c) 2009 Martin O. J. Schmitz.
3     *
4     * This file is part of the SCHMITZM library - a collection of utility
5     * classes based on Java 1.6, focusing (not only) on Java Swing
6     * and the Geotools library.
7     *
8     * The SCHMITZM project is hosted at:
9     * http://wald.intevation.org/projects/schmitzm/
10     *
11     * This program is free software; you can redistribute it and/or
12     * modify it under the terms of the GNU Lesser General Public License
13     * as published by the Free Software Foundation; either version 3
14     * of the License, or (at your option) any later version.
15     *
16     * This program is distributed in the hope that it will be useful,
17     * but WITHOUT ANY WARRANTY; without even the implied warranty of
18     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19     * GNU General Public License for more details.
20     *
21     * You should have received a copy of the GNU Lesser General Public License (license.txt)
22     * along with this program; if not, write to the Free Software
23     * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
24     * or try this link: http://www.gnu.org/licenses/lgpl.html
25     *
26     * Contributors:
27     *     Martin O. J. Schmitz - initial API and implementation
28     *     Stefan A. Krüger - additional utility classes
29     ******************************************************************************/
30  /**  /**
31   Copyright 2008 Stefan Alfons Krüger   Copyright 2008 Stefan Alfons Krüger
32    
33   atlas-framework - This file is part of the Atlas Framework   atlas-framework - This file is part of the Atlas Framework
34    
35   This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version.   This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version.
# Line 13  Line 42 
42   **/   **/
43  package skrueger.geotools.selection;  package skrueger.geotools.selection;
44    
45    import java.awt.RenderingHints;
46  import java.beans.PropertyChangeEvent;  import java.beans.PropertyChangeEvent;
47  import java.beans.PropertyChangeListener;  import java.beans.PropertyChangeListener;
48  import java.util.Arrays;  import java.io.File;
49  import java.util.Collection;  import java.util.HashMap;
50  import java.util.HashSet;  import java.util.HashSet;
51    import java.util.Iterator;
52  import java.util.Set;  import java.util.Set;
53  import java.util.Vector;  import java.util.Vector;
54    
# Line 25  import javax.swing.JTable; Line 56  import javax.swing.JTable;
56  import javax.swing.ListSelectionModel;  import javax.swing.ListSelectionModel;
57  import javax.swing.event.ListSelectionListener;  import javax.swing.event.ListSelectionListener;
58    
59  import org.geotools.feature.Feature;  import org.geotools.feature.FeatureCollection;
60    import org.geotools.feature.FeatureIterator;
61  import org.geotools.map.MapLayer;  import org.geotools.map.MapLayer;
62    import org.geotools.renderer.GTRenderer;
63    import org.geotools.renderer.label.LabelCacheImpl;
64    import org.geotools.renderer.lite.StreamingRenderer;
65  import org.geotools.styling.FeatureTypeStyle;  import org.geotools.styling.FeatureTypeStyle;
66  import org.geotools.styling.Style;  import org.geotools.styling.Style;
67    import org.geotools.styling.visitor.DuplicatingStyleVisitor;
68    import org.opengis.feature.simple.SimpleFeature;
69    import org.opengis.feature.simple.SimpleFeatureType;
70  import org.opengis.filter.identity.FeatureId;  import org.opengis.filter.identity.FeatureId;
71    
72  import schmitzm.geotools.FilterUtil;  import schmitzm.geotools.FilterUtil;
73    import schmitzm.geotools.GTUtil;
74  import schmitzm.geotools.gui.JMapPane;  import schmitzm.geotools.gui.JMapPane;
75  import schmitzm.geotools.map.event.FeatureSelectedEvent;  import schmitzm.geotools.map.event.FeatureSelectedEvent;
76  import schmitzm.geotools.map.event.JMapPaneEvent;  import schmitzm.geotools.map.event.JMapPaneEvent;
77  import schmitzm.geotools.map.event.JMapPaneListener;  import schmitzm.geotools.map.event.JMapPaneListener;
78  import schmitzm.geotools.styling.StylingUtil;  import schmitzm.geotools.styling.StylingUtil;
 import schmitzm.lang.LangUtil;  
79  import skrueger.geotools.MapPaneToolBar;  import skrueger.geotools.MapPaneToolBar;
80  import skrueger.geotools.StyledMapInterface;  import skrueger.geotools.StyledFeaturesInterface;
81    import skrueger.geotools.StyledLayerInterface;
82    
83  /**  /**
84   * This class keeps the selection of a (feature) {@link JTable} synchronized   * This class keeps the selection of a (feature) {@link JTable} synchronized
# Line 68  public class FeatureMapLayerSelectionSyn Line 107  public class FeatureMapLayerSelectionSyn
107           * model.           * model.
108           */           */
109          protected final MapLayer mapLayer;          protected final MapLayer mapLayer;
110          protected final StyledMapInterface<?> styledMapLayer;          protected final StyledFeaturesInterface<?> styledLayer;
111          protected final JMapPane mapPane;          protected final JMapPane mapPane;
112          private final MapPaneToolBar toolBar;          private final MapPaneToolBar toolBar;
113            private final HashMap<Object, Object> defaultGTRenderingHints;
114    
115          /**          /**
116           * Creates a new synchronizer           * Creates a new synchronizer
# Line 84  public class FeatureMapLayerSelectionSyn Line 124  public class FeatureMapLayerSelectionSyn
124           */           */
125          public FeatureMapLayerSelectionSynchronizer(          public FeatureMapLayerSelectionSynchronizer(
126                          StyledFeatureLayerSelectionModel layerSelModel,                          StyledFeatureLayerSelectionModel layerSelModel,
127                          StyledMapInterface<?> styledMapLayer, MapLayer mapLayer,                          StyledFeaturesInterface<?> styledLayer, MapLayer mapLayer,
128                          JMapPane mapPane, MapPaneToolBar toolBar) {                          JMapPane mapPane, MapPaneToolBar toolBar,
129                            HashMap<Object, Object> defaultGTRenderingHints) {
130    
131                  super(layerSelModel);                  super(layerSelModel);
132                  this.styledMapLayer = styledMapLayer;                  this.styledLayer = styledLayer;
133    
134                  this.mapLayer = mapLayer;                  this.mapLayer = mapLayer;
135                  this.mapPane = mapPane;                  this.mapPane = mapPane;
136                  this.toolBar = toolBar;                  this.toolBar = toolBar;
137                    if (defaultGTRenderingHints != null)
138                            this.defaultGTRenderingHints = defaultGTRenderingHints;
139                    else
140                            this.defaultGTRenderingHints = new HashMap<Object, Object>();
141          }          }
142    
143          /**          /**
# Line 138  public class FeatureMapLayerSelectionSyn Line 183  public class FeatureMapLayerSelectionSyn
183           * selection.           * selection.
184           *           *
185           * @param newSelection           * @param newSelection
186           *            A {@link Vector} of Feature-IDs that are selected.           *            A {@link Vector} of SimpleFeature-IDs that are selected.
187           */           */
188          private void changeLayerStyle(final Vector<String> newSelection) {          private void changeLayerStyle(final Vector<String> newSelection) {
                 Style selectionMapStyle = null;  
189                  try {                  try {
190    
191                            Style originalStyle = mapLayer.getStyle();
192                          if (newSelection.isEmpty()) {                          if (newSelection.isEmpty()) {
193    
194                                  selectionMapStyle = styledMapLayer.getStyle();                                  // Check if the Style contains a SELECTION FTS
195                                  // LOGGER.debug("NO SELECTION .. set to original style directly");  
196                                    FeatureTypeStyle[] clone = originalStyle.featureTypeStyles()
197                                                    .toArray(new FeatureTypeStyle[] {}).clone();
198    
199                                    for (FeatureTypeStyle fts : clone) {
200                                            if (fts.getName() != null
201                                                            && fts.getName().equals(SELECTION_STYLING)) {
202                                                    originalStyle.featureTypeStyles().remove(fts);
203    
204                                                    replaceRenderer();
205                                                    // mapPane.refresh();
206    
207                                                    return;
208                                            }
209                                    }
210    
211                          } else {                          } else {
212                                  LOGGER.debug("SELECTION .. change style");                                  LOGGER.debug("SELECTION .. change style");
213    
214                                  // We take Style from the MapLayer that is displayed at the                                  // We take Style from the MapLayer that is displayed at the
215                                  // moment. We do not use the styledMapLayer.getStyle, because in                                  // moment. We do not use the styledLayer.getStyle, because in
216                                  // the atlas, this always return the default style, but                                  // the atlas, this always return the default style, but
217                                  // additionaly styles might be selected.                                  // additional styles might be selected.
218                                  // Taking the style from the mapLayer indicated, that we have to                                  // Taking the style from the mapLayer indicated, that we have to
219                                  // remove any selection rules first.                                  // remove any selection rules first.
                                 Style originalStyle = mapLayer.getStyle();  
220    
221                                  // TODO The default style is not good here. We need                                  FeatureTypeStyle selectionFTStyle = StylingUtil
222                                  // .createSelectionStyle(normalStyle, geoObj);                                                  .createSelectionStyle(styledLayer.getGeoObject());
                                 selectionMapStyle = StylingUtil  
                                                 .createSelectionStyle(styledMapLayer.getGeoObject());  
                                 // selectionMapStyle = StylingUtil  
                                 // .createDefaultStyle(styledMapLayer.getGeoObject());  
223    
224                                  selectionMapStyle.getFeatureTypeStyles()[0]                                  selectionFTStyle.setName(SELECTION_STYLING);
                                                 .setName(SELECTION_STYLING);  
225    
226                                  /**                                  /**
227                                   *                                   *
228                                   * Add a Filter to the selectionMapStyle, so that it is only                                   * Add a Filter to the selectionMapStyle, so that it is only
229                                   * used on objects that are selected. <br/>                                   * used on objects that are selected. <br/>
230                                     * To optimize rendering speed, the filter checks whether more
231                                     * than half of the features are selected. If so, the filter is
232                                     * inverted to be shorter.
233                                   *                                   *
                                  * Note 1:<br/>  
                                  * It is NEVER allowed to GeoTools extend Filter () { .. } (and write  
                                  * tests into the evaluate block). Especially for the  
                                  * ShapeFileRenderer, we may only use a geotools Filter.<br/>  
                                  *  
                                  * Note 2:<br/>  
                                  * The FilterUtil.FILTER_FAC2.id(fids) wants a set of FeatureId-Objects!  
234                                   */                                   */
235    
236                                  Set<FeatureId> fids = new HashSet<FeatureId>();                                  final FeatureCollection<SimpleFeatureType, SimpleFeature> featureCollectionFiltered = styledLayer
237                                  for (String fid : newSelection) {                                                  .getFeatureCollectionFiltered();
238                                          fids.add(FilterUtil.FILTER_FAC2.featureId(fid));                                  if (newSelection.size() > featureCollectionFiltered.size() / 2) {
239                                  }                                          FeatureIterator<SimpleFeature> iterator = featureCollectionFiltered
240                                                            .features();
241                                            Set<FeatureId> antiFids = new HashSet<FeatureId>();
242                                            try {
243                                                    while (iterator.hasNext()) {
244                                                            SimpleFeature next = iterator.next();
245                                                            if (!newSelection.contains(next.getID()))
246                                                                    antiFids.add(FilterUtil.FILTER_FAC2
247                                                                                    .featureId(next.getID()));
248                                                    }
249    
250                                                    selectionFTStyle.rules().get(0).setFilter(
251                                                                    FilterUtil.FILTER_FAC2
252                                                                                    .not(FilterUtil.FILTER_FAC2
253                                                                                                    .id(antiFids)));
254    
255                                            } finally {
256                                                    featureCollectionFiltered.close(iterator);
257                                            }
258                                    } else {
259                                            Set<FeatureId> fids = new HashSet<FeatureId>();
260                                            for (String fid : newSelection) {
261                                                    fids.add(FilterUtil.FILTER_FAC2.featureId(fid));
262                                            }
263    
264                                  selectionMapStyle.getFeatureTypeStyles()[0].getRules()[0]                                          selectionFTStyle.rules().get(0).setFilter(
265                                                  .setFilter(FilterUtil.FILTER_FAC2.id(fids));                                                          FilterUtil.FILTER_FAC2.id(fids));
266                                    }
267    
268                                  FeatureTypeStyle[] originalFeatureTypeStyles = originalStyle                                  // Maybe there has already been another selection
269                                                  .getFeatureTypeStyles();                                  // FeatureTypeStyle... Let's replace it...
270                                  FeatureTypeStyle[] newFeatureTypes;                                  boolean foundAndReplaced = false;
271                                  if (originalFeatureTypeStyles[originalFeatureTypeStyles.length - 1]                                  for (FeatureTypeStyle fts : originalStyle.featureTypeStyles()) {
272                                                  .getName() != null                                          if (fts.getName() != null
273                                                  && originalFeatureTypeStyles[originalFeatureTypeStyles.length - 1]                                                          && fts.getName().equals(SELECTION_STYLING)) {
274                                                                  .getName().equals(SELECTION_STYLING)) {                                                  foundAndReplaced = true;
275                                          newFeatureTypes = Arrays.copyOf(originalFeatureTypeStyles,                                                  fts.rules().clear();
276                                                          originalFeatureTypeStyles.length - 1);                                                  fts.rules().addAll(selectionFTStyle.rules());
277                                  } else {                                                  break;
278                                          newFeatureTypes = originalFeatureTypeStyles;                                          }
279                                    }
280                                    if (!foundAndReplaced) {
281                                            originalStyle.featureTypeStyles().add(selectionFTStyle);
282                                  }                                  }
283    
284                                  // The last FTS is the selection FTS                                  // Refresh the map
285                                  newFeatureTypes = LangUtil.extendArray(newFeatureTypes,                                  // mapPane.refresh();
                                                 selectionMapStyle.getFeatureTypeStyles());  
286    
287                                  selectionMapStyle.setFeatureTypeStyles(newFeatureTypes);                                  DuplicatingStyleVisitor dsv = new DuplicatingStyleVisitor();
288                                    dsv.visit(originalStyle);
289                                    Style newStyle = (Style) dsv.getCopy();
290    
291    //SK-Debug
292    //                              try {
293    ////                                    
294    ////                            StylingUtil.saveStyleToSLD(newStyle, new File(
295    ////                                            "/home/stefan/Desktop/selection.sld"));
296    //                              } catch (Exception e) {
297    //                              }
298    
299                                    // DuplicatingStyleVisitor dsv = new DuplicatingStyleVisitor();
300                                    // dsv.visit(originalStyle);
301                                    // Style newStyle = originalStyle;
302    
303                                    // Style newStyle = StylingUtil.STYLE_BUILDER.createStyle();
304                                    // newStyle.featureTypeStyles().addAll(originalStyle.featureTypeStyles());
305                                    mapLayer.setStyle(newStyle);
306    
307                                    replaceRenderer();
308                          }                          }
309    
                         mapLayer.setStyle(selectionMapStyle);  
                         mapPane.refresh();  
310                  } catch (Exception e) {                  } catch (Exception e) {
311                          LOGGER.error("Error while trying to create a selection style", e);                          LOGGER.error("Error while trying to create a selection style", e);
312                  }                  }
313          }          }
314    
315            private void replaceRenderer() {
316                    //
317                    // // Has to be done before we apply the new Renderer
318                    // mapLayer.setStyle(style);
319    
320                    GTRenderer oldRenderer = mapPane.getRenderer();
321    
322                    /**
323                     * Explicitly putting a new instance of LabelCacheDefault into the
324                     * renderer instance, so JMapPane doesn't reuse the old one. This is
325                     * very useful when changing the TextSymbolizer with AtlasStyler<br/>
326                     * SK 9.7.09: It's not enought to user LabelCache.clear(). We can not
327                     * reuse the old Renderer - better to create a new one!
328                     */
329                    final GTRenderer newRenderer = GTUtil.createGTRenderer();
330    
331                    final HashMap<Object, Object> rendererHints = defaultGTRenderingHints;
332                    rendererHints.put(StreamingRenderer.LABEL_CACHE_KEY,
333                                    new LabelCacheImpl());
334    
335                    newRenderer.setRendererHints(rendererHints);
336                    mapPane.setRenderer(newRenderer);
337    
338                    if (oldRenderer != null) {
339    
340                            RenderingHints java2DHints = oldRenderer.getJava2DHints();
341                            if (java2DHints != null) {
342                                    newRenderer.setJava2DHints(java2DHints);
343                            }
344    
345                            oldRenderer.setContext(null);
346                            oldRenderer = null;
347                    }
348    
349                    mapPane.refresh();
350    
351            }
352    
353          /**          /**
354           * Used to synchronize {@link FeatureSelectedEvent}s with the           * Used to synchronize {@link FeatureSelectedEvent}s with the
355           * {@link StyledFeatureLayerSelectionModel}           * {@link StyledFeatureLayerSelectionModel}
# Line 251  public class FeatureMapLayerSelectionSyn Line 382  public class FeatureMapLayerSelectionSyn
382                  if (e.getSourceObject() != this.mapPane)                  if (e.getSourceObject() != this.mapPane)
383                          return;                          return;
384    
                 FeatureSelectedEvent fse = (FeatureSelectedEvent) e;  
385    
386                  /**                  /**
387                   * Checking, that the FeatureSelectedEvent actually contains features                   * Checking, that the FeatureSelectedEvent actually contains features
388                   * from this layer                   * from this layer
389                   */                   */
390                    FeatureSelectedEvent fse = (FeatureSelectedEvent) e;
391                  final String sourceID = fse.getSourceLayer().getTitle();                  final String sourceID = fse.getSourceLayer().getTitle();
392                  final String syncForID = mapLayer.getTitle();                  final String syncForID = mapLayer.getTitle();
393                  if (sourceID != null && syncForID != null                  if (sourceID != null && syncForID != null
# Line 265  public class FeatureMapLayerSelectionSyn Line 396  public class FeatureMapLayerSelectionSyn
396                          return;                          return;
397                  }                  }
398    
399                  LOGGER.debug("do event " + fse);                  // LOGGER.debug("do event " + fse);
400    
401                  // Avoid event circles in propertyChange(..)                  // Avoid event circles in propertyChange(..)
402                  selectionChangeCausedByMe = true;                  selectionChangeCausedByMe = true;
403    
404                  Collection<Feature> selectionResult = (Collection<Feature>) fse                  final FeatureCollection<SimpleFeatureType, SimpleFeature> selectionResult = fse
405                                  .getSelectionResult();                                  .getSelectionResult();
406                    Iterator<SimpleFeature> fi = selectionResult.iterator();
407                    try {
408    
409                  // reset the selection of the DpLayerSelectionModel                          // reset the selection of the DpLayerSelectionModel
410                  // layerSelModel.setValueIsAdjusting(true);                          // layerSelModel.setValueIsAdjusting(true);
411    
412                  if (selectedTool == MapPaneToolBar.TOOL_SELECTION_ADD) {                          if (selectedTool == MapPaneToolBar.TOOL_SELECTION_ADD) {
413                          layerSelModel.setValueIsAdjusting(true);                                  layerSelModel.setValueIsAdjusting(true);
                         for (Feature f : selectionResult) {  
                                 layerSelModel.addSelection(f.getID());  
                         }  
                         layerSelModel.setValueIsAdjusting(false);  
                 } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_SET) {  
                         layerSelModel.setValueIsAdjusting(true);  
                         layerSelModel.clearSelection();  
414    
415                          for (Feature f : selectionResult) {                                  for (int fIdx = 0; fi.hasNext(); fIdx++) {
416                                  layerSelModel.addSelection(f.getID());                                          SimpleFeature f = fi.next();
417                          }                                          layerSelModel.addSelection(f.getID());
418                                    }
419                                    layerSelModel.setValueIsAdjusting(false);
420                            } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_SET) {
421                                    layerSelModel.setValueIsAdjusting(true);
422                                    layerSelModel.clearSelection();
423    
424                                    for (int fIdx = 0; fi.hasNext(); fIdx++) {
425                                            SimpleFeature f = fi.next();
426                                            layerSelModel.addSelection(f.getID());
427                                    }
428    
429                                    // LOGGER.debug("Setting selection to " + fi.());
430    
431                          LOGGER.debug("Setting selection to " + selectionResult.size());                                  layerSelModel.setValueIsAdjusting(false);
432                          layerSelModel.setValueIsAdjusting(false);                          } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_REMOVE) {
433                  } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_REMOVE) {                                  layerSelModel.setValueIsAdjusting(true);
434                          layerSelModel.setValueIsAdjusting(true);                                  for (int fIdx = 0; fi.hasNext(); fIdx++) {
435                          for (Feature f : selectionResult) {                                          SimpleFeature f = fi.next();
436                                  layerSelModel.removeSelection(f.getID());                                          layerSelModel.removeSelection(f.getID());
437                                    }
438                                    layerSelModel.setValueIsAdjusting(false);
439                          }                          }
440                          layerSelModel.setValueIsAdjusting(false);  
441                    } finally {
442                            selectionResult.close(fi);
443                  }                  }
444    
445                  // Show selected features in the map, which is not automatically done by                  // Show selected features in the map, which is not automatically done by

Legend:
Removed from v.162  
changed lines
  Added in v.493

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26