/[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

revision 427 by alfonx, Fri Oct 2 19:38:44 2009 UTC revision 500 by alfonx, Sun Oct 25 16:56:21 2009 UTC
# Line 45  package skrueger.geotools.selection; Line 45  package skrueger.geotools.selection;
45  import java.awt.RenderingHints;  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.io.File;
49  import java.util.HashMap;  import java.util.HashMap;
50  import java.util.HashSet;  import java.util.HashSet;
51  import java.util.Iterator;  import java.util.Iterator;
# Line 55  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.FeatureCollection;
60    import org.geotools.feature.FeatureIterator;
61    import org.geotools.filter.FidFilterImpl;
62  import org.geotools.map.MapLayer;  import org.geotools.map.MapLayer;
63  import org.geotools.renderer.GTRenderer;  import org.geotools.renderer.GTRenderer;
64  import org.geotools.renderer.label.LabelCacheImpl;  import org.geotools.renderer.label.LabelCacheImpl;
65  import org.geotools.renderer.lite.StreamingRenderer;  import org.geotools.renderer.lite.StreamingRenderer;
 import org.geotools.renderer.shape.ShapefileRenderer;  
66  import org.geotools.styling.FeatureTypeStyle;  import org.geotools.styling.FeatureTypeStyle;
67  import org.geotools.styling.Style;  import org.geotools.styling.Style;
68  import org.geotools.styling.visitor.DuplicatingStyleVisitor;  import org.geotools.styling.visitor.DuplicatingStyleVisitor;
69  import org.opengis.feature.simple.SimpleFeature;  import org.opengis.feature.simple.SimpleFeature;
70    import org.opengis.feature.simple.SimpleFeatureType;
71    import org.opengis.filter.Filter;
72    import org.opengis.filter.Not;
73  import org.opengis.filter.identity.FeatureId;  import org.opengis.filter.identity.FeatureId;
74    
75  import schmitzm.geotools.FilterUtil;  import schmitzm.geotools.FilterUtil;
76    import schmitzm.geotools.GTUtil;
77  import schmitzm.geotools.gui.JMapPane;  import schmitzm.geotools.gui.JMapPane;
78  import schmitzm.geotools.map.event.FeatureSelectedEvent;  import schmitzm.geotools.map.event.FeatureSelectedEvent;
79  import schmitzm.geotools.map.event.JMapPaneEvent;  import schmitzm.geotools.map.event.JMapPaneEvent;
80  import schmitzm.geotools.map.event.JMapPaneListener;  import schmitzm.geotools.map.event.JMapPaneListener;
81  import schmitzm.geotools.styling.StylingUtil;  import schmitzm.geotools.styling.StylingUtil;
82  import skrueger.geotools.MapPaneToolBar;  import skrueger.geotools.MapPaneToolBar;
83  import skrueger.geotools.StyledLayerInterface;  import skrueger.geotools.StyledFeaturesInterface;
84    
85  /**  /**
86   * This class keeps the selection of a (feature) {@link JTable} synchronized   * This class keeps the selection of a (feature) {@link JTable} synchronized
# Line 102  public class FeatureMapLayerSelectionSyn Line 109  public class FeatureMapLayerSelectionSyn
109           * model.           * model.
110           */           */
111          protected final MapLayer mapLayer;          protected final MapLayer mapLayer;
112          protected final StyledLayerInterface<?> styledLayer;          protected final StyledFeaturesInterface<?> styledLayer;
113          protected final JMapPane mapPane;          protected final JMapPane mapPane;
114          private final MapPaneToolBar toolBar;          private final MapPaneToolBar toolBar;
115          private final HashMap<Object, Object> defaultGTRenderingHints;          private final HashMap<Object, Object> defaultGTRenderingHints;
# Line 119  public class FeatureMapLayerSelectionSyn Line 126  public class FeatureMapLayerSelectionSyn
126           */           */
127          public FeatureMapLayerSelectionSynchronizer(          public FeatureMapLayerSelectionSynchronizer(
128                          StyledFeatureLayerSelectionModel layerSelModel,                          StyledFeatureLayerSelectionModel layerSelModel,
129                          StyledLayerInterface<?> styledLayer, MapLayer mapLayer,                          StyledFeaturesInterface<?> styledLayer, MapLayer mapLayer,
130                          JMapPane mapPane, MapPaneToolBar toolBar, HashMap<Object, Object> defaultGTRenderingHints) {                          JMapPane mapPane, MapPaneToolBar toolBar,
131                            HashMap<Object, Object> defaultGTRenderingHints) {
132    
133                  super(layerSelModel);                  super(layerSelModel);
134                  this.styledLayer = styledLayer;                  this.styledLayer = styledLayer;
# Line 130  public class FeatureMapLayerSelectionSyn Line 138  public class FeatureMapLayerSelectionSyn
138                  this.toolBar = toolBar;                  this.toolBar = toolBar;
139                  if (defaultGTRenderingHints != null)                  if (defaultGTRenderingHints != null)
140                          this.defaultGTRenderingHints = defaultGTRenderingHints;                          this.defaultGTRenderingHints = defaultGTRenderingHints;
141                  else                  else
142                          this.defaultGTRenderingHints = new HashMap<Object, Object>();                          this.defaultGTRenderingHints = new HashMap<Object, Object>();
143          }          }
144    
# Line 181  public class FeatureMapLayerSelectionSyn Line 189  public class FeatureMapLayerSelectionSyn
189           */           */
190          private void changeLayerStyle(final Vector<String> newSelection) {          private void changeLayerStyle(final Vector<String> newSelection) {
191                  try {                  try {
 //                      GTRenderer r = mapPane.getRenderer();  
 //                      if (r instanceof ShapefileRenderer) {  
 //                              ShapefileRenderer sfr = (ShapefileRenderer)r;  
 //                              sfr.setCaching(false);  
 //                      }  
 //                        
                           
192    
                           
193                          Style originalStyle = mapLayer.getStyle();                          Style originalStyle = mapLayer.getStyle();
194    
195                          if (newSelection.isEmpty()) {                          if (newSelection.isEmpty()) {
196    
197                                  // Check if the Style contains a SELECTION FTS                                  // Check if the Style contains a SELECTION FTS
198                                    
199                                  FeatureTypeStyle[] clone = originalStyle.featureTypeStyles().toArray( new FeatureTypeStyle[] {} ).clone();                                  FeatureTypeStyle[] clone = originalStyle.featureTypeStyles()
200                                                                                    .toArray(new FeatureTypeStyle[] {}).clone();
201    
202                                  for (FeatureTypeStyle fts : clone) {                                  for (FeatureTypeStyle fts : clone) {
203                                          if (fts.getName() != null                                          if (fts.getName() != null
204                                                          && fts.getName().equals(SELECTION_STYLING)) {                                                          && fts.getName().equals(SELECTION_STYLING)) {
205                                                  originalStyle.featureTypeStyles().remove(fts);                                                  originalStyle.featureTypeStyles().remove(fts);
206                                                    
207                                                  replaceRenderer();                                                  replaceRenderer();
208  //                                              mapPane.refresh();                                                  // mapPane.refresh();
                                                   
209    
210                                                  return;                                                  return;
211                                          }                                          }
212                                  }                                  }
213    
214                          } else {                          } else {
215    
216                                  LOGGER.debug("SELECTION .. change style");                                  LOGGER.debug("SELECTION .. change style");
217    
218                                    // Saving a repaint if the selection didn't change
219                                    if (!selectionChanged(newSelection, originalStyle)) {
220                                            return;
221                                    }
222    
223                                  // We take Style from the MapLayer that is displayed at the                                  // We take Style from the MapLayer that is displayed at the
224                                  // moment. We do not use the styledLayer.getStyle, because in                                  // moment. We do not use the styledLayer.getStyle, because in
225                                  // the atlas, this always return the default style, but                                  // the atlas, this always return the default style, but
# Line 229  public class FeatureMapLayerSelectionSyn Line 236  public class FeatureMapLayerSelectionSyn
236                                   *                                   *
237                                   * Add a Filter to the selectionMapStyle, so that it is only                                   * Add a Filter to the selectionMapStyle, so that it is only
238                                   * used on objects that are selected. <br/>                                   * used on objects that are selected. <br/>
239                                     * To optimize rendering speed, the filter checks whether more
240                                     * than half of the features are selected. If so, the filter is
241                                     * inverted to be shorter.
242                                   *                                   *
                                  * 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!  
243                                   */                                   */
244                                    final FeatureCollection<SimpleFeatureType, SimpleFeature> featureCollectionFiltered = styledLayer
245                                                    .getFeatureCollectionFiltered();
246                                    if (newSelection.size() > featureCollectionFiltered.size() / 2) {
247                                            FeatureIterator<SimpleFeature> iterator = featureCollectionFiltered
248                                                            .features();
249                                            Set<FeatureId> antiFids = new HashSet<FeatureId>();
250                                            try {
251                                                    while (iterator.hasNext()) {
252                                                            SimpleFeature next = iterator.next();
253                                                            if (!newSelection.contains(next.getID()))
254                                                                    antiFids.add(FilterUtil.FILTER_FAC2
255                                                                                    .featureId(next.getID()));
256                                                    }
257    
258                                                    selectionFTStyle.rules().get(0).setFilter(
259                                                                    FilterUtil.FILTER_FAC2
260                                                                                    .not(FilterUtil.FILTER_FAC2
261                                                                                                    .id(antiFids)));
262    
263                                  Set<FeatureId> fids = new HashSet<FeatureId>();                                          } finally {
264                                  for (String fid : newSelection) {                                                  featureCollectionFiltered.close(iterator);
265                                          fids.add(FilterUtil.FILTER_FAC2.featureId(fid));                                          }
266                                  }                                  } else {
267                                            Set<FeatureId> fids = new HashSet<FeatureId>();
268                                            for (String fid : newSelection) {
269                                                    fids.add(FilterUtil.FILTER_FAC2.featureId(fid));
270                                            }
271    
272                                  selectionFTStyle.rules().get(0).setFilter(                                          selectionFTStyle.rules().get(0).setFilter(
273                                                  FilterUtil.FILTER_FAC2.id(fids));                                                          FilterUtil.FILTER_FAC2.id(fids));
274                                    }
275    
276                                  // Maybe there has already been another selection                                  // Maybe there has already been another selection
277                                  // FeatureTypeStyle... Let's replace it...                                  // FeatureTypeStyle... Let's replace it...
# Line 263  public class FeatureMapLayerSelectionSyn Line 288  public class FeatureMapLayerSelectionSyn
288                                  if (!foundAndReplaced) {                                  if (!foundAndReplaced) {
289                                          originalStyle.featureTypeStyles().add(selectionFTStyle);                                          originalStyle.featureTypeStyles().add(selectionFTStyle);
290                                  }                                  }
                                   
                                   
291    
292                                  // Refresh the map                                  // Refresh the map
293  //                              mapPane.refresh();                                  // mapPane.refresh();
294                                    
295                                  DuplicatingStyleVisitor dsv = new DuplicatingStyleVisitor();                                  DuplicatingStyleVisitor dsv = new DuplicatingStyleVisitor();
296                                  dsv.visit(originalStyle);                                  dsv.visit(originalStyle);
297                                  Style newStyle = (Style)dsv.getCopy();                                  Style newStyle = (Style) dsv.getCopy();
298                                    
299  //                              Style newStyle = StylingUtil.STYLE_BUILDER.createStyle();                                  // SK-Debug
300  //                              newStyle.featureTypeStyles().addAll(originalStyle.featureTypeStyles());                                  try {
301                                            //
302                                            StylingUtil.saveStyleToSLD(newStyle, new File(
303                                                            "/home/stefan/Desktop/selection.sld"));
304                                    } catch (Exception e) {
305                                    }
306    
307                                    // DuplicatingStyleVisitor dsv = new DuplicatingStyleVisitor();
308                                    // dsv.visit(originalStyle);
309                                    // Style newStyle = originalStyle;
310    
311                                    // Style newStyle = StylingUtil.STYLE_BUILDER.createStyle();
312                                    // newStyle.featureTypeStyles().addAll(originalStyle.featureTypeStyles());
313                                  mapLayer.setStyle(newStyle);                                  mapLayer.setStyle(newStyle);
314                                    
315                                  replaceRenderer();                                  replaceRenderer();
316                          }                          }
 //                        
 //                      if (r instanceof ShapefileRenderer) {  
 //                              ShapefileRenderer sfr = (ShapefileRenderer)r;  
 //                              sfr.setCaching(true);  
 //                      }  
   
317    
318                  } catch (Exception e) {                  } catch (Exception e) {
319                          LOGGER.error("Error while trying to create a selection style", e);                          LOGGER.error("Error while trying to create a selection style", e);
320                  }                  }
321          }          }
322    
323            /**
324             * Analyses whether the selection has changed in comparison to the selection
325             * stored in the mapLayer.Style
326             *
327             * @param newSelection
328             * @param originalStyle
329             * @return
330             */
331            private boolean selectionChanged(Vector<String> newSelection,
332                            Style originalStyle) {
333    
334                    boolean SELECTION_STYLING_foundInMapStyle = false;
335    
336                    /**
337                     * testing, whether the selection really changed. If not, we can save
338                     * one paint!
339                     */
340                    for (FeatureTypeStyle fts : originalStyle.featureTypeStyles()) {
341    
342                            if (fts.getName() != null
343                                            && fts.getName().equals(SELECTION_STYLING)) {
344    
345                                    SELECTION_STYLING_foundInMapStyle = true;
346    
347                                    Filter filter = fts.rules().get(0).getFilter();
348                                    if (filter instanceof Not) {
349                                            FidFilterImpl antiOrigFidsFilter = (FidFilterImpl) ((Not) filter)
350                                                            .getFilter();
351    
352                                            // Check one way
353                                            final Set<String> antiFids = antiOrigFidsFilter
354                                                            .getFidsSet();
355    
356                                            if (antiFids.isEmpty() && !newSelection.isEmpty())
357                                                    return true;
358    
359                                            for (String fid : antiFids) {
360                                                    if (newSelection.contains(fid)) {
361                                                            return true;
362                                                    }
363                                            }
364    
365                                            // Check the other way
366                                            for (String fid : newSelection) {
367                                                    if (antiFids.contains(fid)) {
368                                                            return true;
369                                                    }
370                                            }
371    
372                                    } else {
373                                            FidFilterImpl origFidsFilter = (FidFilterImpl) filter;
374    
375                                            // Check one way
376                                            final Set<String> fids = origFidsFilter.getFidsSet();
377    
378                                            if (fids.isEmpty() && newSelection.isEmpty())
379                                                    return false;
380                                            if (fids.size() != newSelection.size())
381                                                    return true;
382    
383                                            for (String fid : fids) {
384                                                    if (!newSelection.contains(fid)) {
385                                                            return true;
386                                                    }
387                                            }
388    
389                                            // Check the other way
390                                            for (String fid : newSelection) {
391                                                    if (!fids.contains(fid)) {
392                                                            return true;
393                                                    }
394                                            }
395    
396                                    }
397    
398                                    break;
399                            }
400                    }
401    
402                    if (!SELECTION_STYLING_foundInMapStyle && !newSelection.isEmpty())
403                            return true;
404    
405                    return false;
406            }
407    
408          private void replaceRenderer() {          private void replaceRenderer() {
409  //                  //
410  //              // Has to be done before we apply the new Renderer                  // // Has to be done before we apply the new Renderer
411  //              mapLayer.setStyle(style);                  // mapLayer.setStyle(style);
412    
413                  GTRenderer oldRenderer = mapPane.getRenderer();                  GTRenderer oldRenderer = mapPane.getRenderer();
414    
# Line 305  public class FeatureMapLayerSelectionSyn Line 419  public class FeatureMapLayerSelectionSyn
419                   * SK 9.7.09: It's not enought to user LabelCache.clear(). We can not                   * SK 9.7.09: It's not enought to user LabelCache.clear(). We can not
420                   * reuse the old Renderer - better to create a new one!                   * reuse the old Renderer - better to create a new one!
421                   */                   */
422                  final GTRenderer newRenderer = new ShapefileRenderer();                  final GTRenderer newRenderer = GTUtil.createGTRenderer();
423    
424                  final HashMap<Object, Object> rendererHints = defaultGTRenderingHints;                  final HashMap<Object, Object> rendererHints = defaultGTRenderingHints;
425                  rendererHints.put(StreamingRenderer.LABEL_CACHE_KEY,                  rendererHints.put(StreamingRenderer.LABEL_CACHE_KEY,
# Line 315  public class FeatureMapLayerSelectionSyn Line 429  public class FeatureMapLayerSelectionSyn
429                  mapPane.setRenderer(newRenderer);                  mapPane.setRenderer(newRenderer);
430    
431                  if (oldRenderer != null) {                  if (oldRenderer != null) {
432                            
433                          RenderingHints java2DHints = oldRenderer.getJava2DHints();                          RenderingHints java2DHints = oldRenderer.getJava2DHints();
434                          if (java2DHints != null) {                          if (java2DHints != null) {
435                                  newRenderer.setJava2DHints(java2DHints);                                  newRenderer.setJava2DHints(java2DHints);
436                          }                          }
437                            
438                          oldRenderer.setContext(null);                          oldRenderer.setContext(null);
439                          oldRenderer = null;                          oldRenderer = null;
440                  }                  }
441                    
442                  mapPane.refresh();                  mapPane.refresh();
443    
444          }          }
# Line 361  public class FeatureMapLayerSelectionSyn Line 475  public class FeatureMapLayerSelectionSyn
475                  if (e.getSourceObject() != this.mapPane)                  if (e.getSourceObject() != this.mapPane)
476                          return;                          return;
477    
                 FeatureSelectedEvent fse = (FeatureSelectedEvent) e;  
   
478                  /**                  /**
479                   * Checking, that the FeatureSelectedEvent actually contains features                   * Checking, that the FeatureSelectedEvent actually contains features
480                   * from this layer                   * from this layer
481                   */                   */
482                    FeatureSelectedEvent fse = (FeatureSelectedEvent) e;
483                  final String sourceID = fse.getSourceLayer().getTitle();                  final String sourceID = fse.getSourceLayer().getTitle();
484                  final String syncForID = mapLayer.getTitle();                  final String syncForID = mapLayer.getTitle();
485                  if (sourceID != null && syncForID != null                  if (sourceID != null && syncForID != null
# Line 375  public class FeatureMapLayerSelectionSyn Line 488  public class FeatureMapLayerSelectionSyn
488                          return;                          return;
489                  }                  }
490    
491                  LOGGER.debug("do event " + fse);                  // LOGGER.debug("do event " + fse);
492    
493                  // Avoid event circles in propertyChange(..)                  // Avoid event circles in propertyChange(..)
494                  selectionChangeCausedByMe = true;                  selectionChangeCausedByMe = true;
495    
496                  Iterator<SimpleFeature> fi = fse.getSelectionResult().iterator();                  final FeatureCollection<SimpleFeatureType, SimpleFeature> selectionResult = fse
497                                    .getSelectionResult();
498                    Iterator<SimpleFeature> fi = selectionResult.iterator();
499                    try {
500    
501                  // reset the selection of the DpLayerSelectionModel                          // reset the selection of the DpLayerSelectionModel
502                  // layerSelModel.setValueIsAdjusting(true);                          // layerSelModel.setValueIsAdjusting(true);
503    
504                  if (selectedTool == MapPaneToolBar.TOOL_SELECTION_ADD) {                          if (selectedTool == MapPaneToolBar.TOOL_SELECTION_ADD) {
505                          layerSelModel.setValueIsAdjusting(true);                                  layerSelModel.setValueIsAdjusting(true);
506    
507                          for (int fIdx = 0; fi.hasNext(); fIdx++) {                                  for (int fIdx = 0; fi.hasNext(); fIdx++) {
508                                  SimpleFeature f = fi.next();                                          SimpleFeature f = fi.next();
509                                  layerSelModel.addSelection(f.getID());                                          layerSelModel.addSelection(f.getID());
510                          }                                  }
511                          layerSelModel.setValueIsAdjusting(false);                                  layerSelModel.setValueIsAdjusting(false);
512                  } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_SET) {                          } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_SET) {
513                          layerSelModel.setValueIsAdjusting(true);                                  layerSelModel.setValueIsAdjusting(true);
514                          layerSelModel.clearSelection();                                  layerSelModel.clearSelection();
515    
516                          for (int fIdx = 0; fi.hasNext(); fIdx++) {                                  for (int fIdx = 0; fi.hasNext(); fIdx++) {
517                                  SimpleFeature f = fi.next();                                          SimpleFeature f = fi.next();
518                                  layerSelModel.addSelection(f.getID());                                          layerSelModel.addSelection(f.getID());
519                          }                                  }
520    
521                          // LOGGER.debug("Setting selection to " + fi.());                                  // LOGGER.debug("Setting selection to " + fi.());
522    
523                          layerSelModel.setValueIsAdjusting(false);                                  layerSelModel.setValueIsAdjusting(false);
524                  } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_REMOVE) {                          } else if (selectedTool == MapPaneToolBar.TOOL_SELECTION_REMOVE) {
525                          layerSelModel.setValueIsAdjusting(true);                                  layerSelModel.setValueIsAdjusting(true);
526                          for (int fIdx = 0; fi.hasNext(); fIdx++) {                                  for (int fIdx = 0; fi.hasNext(); fIdx++) {
527                                  SimpleFeature f = fi.next();                                          SimpleFeature f = fi.next();
528                                  layerSelModel.removeSelection(f.getID());                                          layerSelModel.removeSelection(f.getID());
529                                    }
530                                    layerSelModel.setValueIsAdjusting(false);
531                          }                          }
532                          layerSelModel.setValueIsAdjusting(false);  
533                    } finally {
534                            selectionResult.close(fi);
535                  }                  }
536    
537                  // 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.427  
changed lines
  Added in v.500

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26