source: osm/applications/editors/josm/plugins/cadastre-fr/src/cadastre_fr/CadastrePlugin.java@ 25046

Last change on this file since 25046 was 25046, checked in by pieren, 14 years ago

update comments

  • Property svn:eol-style set to native
File size: 19.7 KB
Line 
1// License: GPL. v2 and later. Copyright 2008-2009 by Pieren <pieren3@gmail.com> and others
2package cadastre_fr;
3
4import static org.openstreetmap.josm.gui.help.HelpUtil.ht;
5import static org.openstreetmap.josm.tools.I18n.marktr;
6import static org.openstreetmap.josm.tools.I18n.tr;
7
8import java.awt.event.ActionEvent;
9import java.awt.event.ActionListener;
10import java.awt.event.KeyEvent;
11import java.io.File;
12
13import javax.swing.JCheckBoxMenuItem;
14import javax.swing.JDialog;
15import javax.swing.JMenu;
16import javax.swing.JMenuItem;
17import javax.swing.JOptionPane;
18import javax.swing.KeyStroke;
19
20import org.openstreetmap.josm.Main;
21import org.openstreetmap.josm.actions.JosmAction;
22import org.openstreetmap.josm.actions.UploadAction;
23import org.openstreetmap.josm.gui.MainMenu;
24import org.openstreetmap.josm.gui.MapFrame;
25import org.openstreetmap.josm.gui.IconToggleButton;
26import org.openstreetmap.josm.gui.layer.Layer;
27import org.openstreetmap.josm.gui.preferences.PreferenceSetting;
28import org.openstreetmap.josm.plugins.Plugin;
29import org.openstreetmap.josm.plugins.PluginInformation;
30import org.openstreetmap.josm.data.projection.*;
31
32/**
33 *
34 * Plugin to access the French Cadastre WMS server at www.cadastre.gouv.fr This
35 * WMS server requires some specific handling like retrieving a cookie for a
36 * limitation in case of no activity, or the request to the server shall provide
37 * a city/town/village code.
38 *
39 * @author Pieren <pieren3@gmail.com>,
40 * @author <matthieu.lochegnies@gmail.com> for the extension to codeCommune
41 * @version 0.8
42 * History:
43 * 0.1 17-Jun-2008 first prototype using a first Lambert projection impl. in core
44 * 0.2 22-Jun-2008 first stable version
45 * 0.3 24-Jun-2008 add code departement
46 * 0.4 06-Jul-2008 - add images scales, icons, menu items disabling
47 * - remove dependencies of wmsplugin
48 * - add option to force a Lambert zone (for median locations)
49 * - add auto-sourcing
50 * 0.5 16-Aug-2008 - add transparency in layer (allowing multiple wms layers displayed together)
51 * - no overlapping of grabbed images if transparency is enabled
52 * - set one layer per location
53 * - use utf-8 charset in POST request to server
54 * - improve the preferences setting dialog
55 * - cancel the current download is now possible
56 * - add automatic images caching and load on request (+ manage cache directory size)
57 * - enable auto-sourcing only if a WMS layer is used
58 * 0.6 18-Aug-2008 - suppress the null-exception message after the dialog 'open a layer first'
59 * - process the overlapping images when cache is loaded from disk
60 * - save the last 'new location request' text again in preferences
61 * - avoid duplicate layers with same name
62 * - set text input for new locations in upper case
63 * - the cache directory is configurable in "cadastrewms.cacheDir"
64 * - improve configuration change updates
65 * 0.7 24-Aug-2008 - mask images only if transparency enabled
66 * - validate projection name by Lambert.toString() method
67 * 0.8 25-Jan-2009 - display returned list of communes if direct name is not recognized by server
68 * - new possible grab factor of 100 square meters fixed size
69 * - minor fixes due to changes in JOSM core classes
70 * - first draft of raster image support
71 * 0.9 05-Feb-2009 - grab vectorized full commune bbox, save in file, convert to OSM way
72 * and simplify
73 * 1.0 18-Feb-2009 - fix various bugs in color management and preference dialog
74 * - increase PNG picture size requested to WMS (800x1000)
75 * - set 4th grab scale fixed size configurable (from 25 to 1000 meters)
76 * 1.1 11-Jun-2009 - fixed a null exception error when trying to displace a vectorized layer
77 * - propose to use shortcut F11 for grabbing
78 * 1.2 16-Aug-2009 - implementation of raster image grabbing, cropping and georeferencing (not the
79 * overview rasters (Tableau d'assemblage) but directly small units (Feuille)
80 * 1.3 23-Aug-2009 - improve georeferencing action cancellation
81 * - fixed bug of raster image loaded from cache not working on Java1.6
82 * - improve mouse click bounce detection during georeferencing process
83 * 1.4 25-Oct-2009 - add support for new Lambert CC 9 Zones projection
84 * - add optional crosspieces display on raster image layers
85 * - add automatic raster images georeferencing when WMS provides data
86 * - re-implement manual adjustment mode in raster image layer
87 * 1.5 21-Nov-2009 - major changes in projection in core : no magical zone prediction anymore for
88 * Lambert 4 and 9 zones; grid translation implemented for Lambert 4 zones;
89 * support of subprojections in preferences for zones setting and UTM20N
90 * - removed autosourcing of empty new nodes
91 * 1.6 28-Nov-2009 - Fix minor issues if Grab is called without layer (possible since projection rework)
92 * 1.7 12-Dec-2009 - Change URL's changes for cookie and downgrade imgs resolution due to WMS changes
93 * 1.8 11-Mar-2010 - filter the mouse button 1 during georeferencing
94 * - retry if getting a new cookie failed (10 times during 30 seconds)
95 * - cookie expiration automatically detected and renewed (after 30 minutes)
96 * - proper WMS layer cleanup at destruction (workaround for memory leak)
97 * - new cache format (v3) storing original image and cropped image bbox + angle
98 * - new cache format (v4) storing original image size for later rotation
99 * - cache files read compatible with previous formats
100 * - raster image rotation issues fixed, now using shift+ctrl key instead of ctrl
101 * - raster image adjustment using default system menu modifier (ctrl for windows) for Mac support
102 * - image resolution configurable (high, medium, low) like the online interface
103 * - layer selection configurable for vectorized images
104 * - improved download cancellation
105 * - from Erik Amzallag:
106 * - possibility to modify the auto-sourcing text just before upload
107 * - from Clément Ménier:
108 * - new option allowing an auto-selection of the first cadastre layer for grab
109 * - non-modal JDialog in MenuActionGrabPlanImage
110 * - new options in the image filter (bilinear, bicubic)
111 * 1.9 05-Apr-2010 - added a scroll bar in preferences
112 * - download cancellation improved
113 * - last deployment for Java1.5 compatibility
114 * 2.0 07-Jul-2010 - update projection for "La Reunion" departement to RGR92, UTM40S.
115 * - add 'departement' as option in the municipality selection
116 * - fixed bug in cache directory size control (and disabled by default)
117 * - add map mode for addressing
118 * - from Nicolas Dumoulin:
119 * - add "tableau d'assemblage" in raster images for georeferencing (as option)
120 * 2.1 14-Jan-2011 - add GrabThread moving the grab to a separate thread
121 * - the divided BBox mode starts from the central square and loads the next in a spiral
122 * - move the grabber from CadastrPlugin singleton to each wmsLayer instance to allow grabbing
123 * of multiple municipalities in parallel.
124 */
125public class CadastrePlugin extends Plugin {
126 static String VERSION = "2.1";
127
128 static JMenu cadastreJMenu;
129
130 public static String source = "";
131
132 // true if the checkbox "auto-sourcing" is set in the plugin menu
133 public static boolean autoSourcing = false;
134
135 // true when the plugin is first used, e.g. grab from WMS or download cache file
136 public static boolean pluginUsed = false;
137
138 public static String cacheDir = null;
139
140 public static boolean alterColors = false;
141
142 public static boolean backgroundTransparent = false;
143
144 public static float transparency = 1.0f;
145
146 public static boolean drawBoundaries = false;
147
148 public static int imageWidth, imageHeight;
149
150 public static String grabLayers, grabStyles = null;
151
152 static private boolean menuEnabled = false;
153
154 /**
155 * Creates the plugin and setup the default settings if necessary
156 *
157 * @throws Exception
158 */
159 public CadastrePlugin(PluginInformation info) throws Exception {
160 super(info);
161 System.out.println("Pluging cadastre-fr v"+VERSION+" started...");
162 if (Main.pref.get("cadastrewms.cacheDir").equals(""))
163 cacheDir = Main.pref.getPreferencesDir()+"plugins"+File.separatorChar+"cadastrewms"+File.separatorChar;
164 else {
165 cacheDir = Main.pref.get("cadastrewms.cacheDir");
166 if (cacheDir.charAt(cacheDir.length()-1) != File.separatorChar )
167 cacheDir += File.separatorChar;
168 }
169 System.out.println("current cache directory: "+cacheDir);
170
171 refreshConfiguration();
172
173 UploadAction.registerUploadHook(new CheckSourceUploadHook());
174
175 }
176
177 public static void refreshMenu() {
178 MainMenu menu = Main.main.menu;
179
180 if (cadastreJMenu == null) {
181 cadastreJMenu = menu.addMenu(marktr("Cadastre"), KeyEvent.VK_C, menu.defaultMenuPos, ht("/Plugin/CadastreFr"));
182 JosmAction grab = new MenuActionGrab();
183 JMenuItem menuGrab = new JMenuItem(grab);
184 KeyStroke ks = grab.getShortcut().getKeyStroke();
185 if (ks != null) {
186 menuGrab.setAccelerator(ks);
187 }
188 JMenuItem menuActionGrabPlanImage = new JMenuItem(new MenuActionGrabPlanImage());
189 JMenuItem menuSettings = new JMenuItem(new MenuActionNewLocation());
190 final JCheckBoxMenuItem menuSource = new JCheckBoxMenuItem(tr("Auto sourcing"));
191 menuSource.setSelected(autoSourcing);
192 menuSource.addActionListener(new ActionListener() {
193 public void actionPerformed(ActionEvent ev) {
194 Main.pref.put("cadastrewms.autosourcing", menuSource.isSelected());
195 autoSourcing = menuSource.isSelected();
196 }
197 });
198
199 //JMenuItem menuResetCookie = new JMenuItem(new MenuActionResetCookie());
200 //JMenuItem menuLambertZone = new JMenuItem(new MenuActionLambertZone());
201 JMenuItem menuLoadFromCache = new JMenuItem(new MenuActionLoadFromCache());
202 // temporary disabled:
203 //JMenuItem menuActionBoundaries = new JMenuItem(new MenuActionBoundaries());
204 //JMenuItem menuActionBuildings = new JMenuItem(new MenuActionBuildings());
205
206 cadastreJMenu.add(menuGrab);
207 cadastreJMenu.add(menuActionGrabPlanImage);
208 cadastreJMenu.add(menuSettings);
209 cadastreJMenu.add(menuSource);
210 //cadastreJMenu.add(menuResetCookie); not required any more
211 //cadastreJMenu.add(menuLambertZone);
212 //if (Main.pref.getBoolean("cadastrewms.buildingsMenu", false))
213 // cadastreJMenu.add(menuActionBuildings);
214 cadastreJMenu.add(menuLoadFromCache);
215 // all SVG features disabled until official WMS is released
216 //cadastreJMenu.add(menuActionBoundaries);
217 }
218 setEnabledAll(menuEnabled);
219 }
220
221 public static void refreshConfiguration() {
222 source = checkSourceMillesime();
223 autoSourcing = Main.pref.getBoolean("cadastrewms.autosourcing", true);
224 alterColors = Main.pref.getBoolean("cadastrewms.alterColors");
225 drawBoundaries = Main.pref.getBoolean("cadastrewms.drawBoundaries", false);
226 if (alterColors) {
227 backgroundTransparent = Main.pref.getBoolean("cadastrewms.backgroundTransparent");
228 transparency = Float.parseFloat(Main.pref.get("cadastrewms.brightness", "1.0f"));
229 } else {
230 backgroundTransparent = false;
231 transparency = 1.0f;
232 }
233 String currentResolution = Main.pref.get("cadastrewms.resolution", "high");
234 if (currentResolution.equals("high")) {
235 imageWidth = 1000; imageHeight = 800;
236 } else if (currentResolution.equals("medium")){
237 imageWidth = 800; imageHeight = 600;
238 } else {
239 imageWidth = 600; imageHeight = 400;
240 }
241 refreshLayersURL();
242
243 // overwrite F11 shortcut used from the beginning by this plugin and recently used
244 // for full-screen switch in JOSM core
245 int i = 0;
246 String p = Main.pref.get("shortcut.shortcut."+i, null);
247 boolean alreadyRedefined = false;
248 while (p != null) {
249 String[] s = p.split(";");
250 alreadyRedefined = alreadyRedefined || s[0].equals("menu:view:fullscreen");
251 i++;
252 p = Main.pref.get("shortcut.shortcut."+i, null);
253 }
254 if (!alreadyRedefined) {
255 int reply = JOptionPane.showConfirmDialog(null,
256 tr("Plugin cadastre-fr used traditionaly for grabbing the key shortcut F11\n"+
257 "which is currently allocated for full-screen switch by default\n"+
258 "Would you like to restore F11 for grab action ?"),
259 tr("Restore grab shortcut F11"),
260 JOptionPane.YES_NO_OPTION);
261 if (reply == JOptionPane.OK_OPTION) {
262 System.out.println("redefine fullscreen shortcut F11 to shift+F11");
263 Main.pref.put("shortcut.shortcut."+i, "menu:view:fullscreen;Toggle Full Screen view;122;5;122;64;false;true");
264 JOptionPane.showMessageDialog(Main.parent,tr("JOSM is stopped for the change to take effect."));
265 System.exit(0);
266 }
267 } else
268 System.out.println("shortcut F11 already redefined; do not change");
269
270 refreshMenu();
271 }
272
273 private static void refreshLayersURL() {
274 grabLayers = "";
275 grabStyles = "";
276 if (Main.pref.getBoolean("cadastrewms.layerWater", true)) {
277 grabLayers += "CDIF:LS3,";
278 grabStyles += "LS3_90,";
279 }
280 if (Main.pref.getBoolean("cadastrewms.layerBuilding", true)) {
281 grabLayers += "CDIF:LS2,";
282 grabStyles += "LS2_90,";
283 }
284 if (Main.pref.getBoolean("cadastrewms.layerSymbol", true)) {
285 grabLayers += "CDIF:LS1,";
286 grabStyles += "LS1_90,";
287 }
288 if (Main.pref.getBoolean("cadastrewms.layerParcel", true)) {
289 grabLayers += "CDIF:PARCELLE,";
290 grabStyles += "PARCELLE_90,";
291 }
292 if (Main.pref.getBoolean("cadastrewms.layerNumero", true)) {
293 grabLayers += "CDIF:NUMERO,";
294 grabStyles += "NUMERO_90,";
295 }
296 if (Main.pref.getBoolean("cadastrewms.layerLabel", true)) {
297 grabLayers += "CDIF:PT3,CDIF:PT2,CDIF:PT1,";
298 grabStyles += "PT3_90,PT2_90,PT1_90,";
299 }
300 if (Main.pref.getBoolean("cadastrewms.layerLieudit", true)) {
301 grabLayers += "CDIF:LIEUDIT,";
302 grabStyles += "LIEUDIT_90,";
303 }
304 if (Main.pref.getBoolean("cadastrewms.layerSection", true)) {
305 grabLayers += "CDIF:SUBSECTION,CDIF:SECTION,";
306 grabStyles += "SUBSECTION_90,SECTION_90,";
307 }
308 if (Main.pref.getBoolean("cadastrewms.layerCommune", true)) {
309 grabLayers += "CDIF:COMMUNE,";
310 grabStyles += "COMMUNE_90,";
311 }
312 if (grabLayers.length() > 0) { // remove the last ','
313 grabLayers = grabLayers.substring(0, grabLayers.length()-1);
314 grabStyles = grabStyles.substring(0, grabStyles.length()-1);
315 }
316 }
317
318 @Override
319 public PreferenceSetting getPreferenceSetting() {
320 return new CadastrePreferenceSetting();
321 }
322
323 private static void setEnabledAll(boolean isEnabled) {
324 for (int i = 0; i < cadastreJMenu.getItemCount(); i++) {
325 JMenuItem item = cadastreJMenu.getItem(i);
326 if (item != null)
327 if (item.getText().equals(MenuActionGrabPlanImage.name) /*||
328 item.getText().equals(MenuActionGrab.name) ||
329 item.getText().equals(MenuActionBoundaries.name) ||
330 item.getText().equals(MenuActionBuildings.name)*/) {
331 item.setEnabled(isEnabled);
332 }
333 }
334 menuEnabled = isEnabled;
335 }
336
337 public void mapFrameInitialized(MapFrame oldFrame, MapFrame newFrame) {
338 if (cadastreJMenu != null) {
339 if (oldFrame == null && newFrame != null) {
340 setEnabledAll(true);
341 Main.map.addMapMode(new IconToggleButton(new WMSAdjustAction(Main.map)));
342 Main.map.addMapMode(new IconToggleButton(new Address(Main.map)));
343 } else if (oldFrame != null && newFrame == null) {
344 setEnabledAll(false);
345 //Lambert.layoutZone = -1;
346 //LambertCC9Zones.layoutZone = -1;
347 }
348 }
349 }
350
351 public static boolean isCadastreProjection() {
352 return Main.proj.toString().equals(new Lambert().toString())
353 || Main.proj.toString().equals(new UTM_France_DOM().toString())
354 || Main.proj.toString().equals(new LambertCC9Zones().toString());
355 }
356
357 public static void safeSleep(long milliseconds) {
358 try {
359 Thread.sleep(milliseconds);
360 } catch (InterruptedException e) {}
361 }
362
363 // See OptionPaneUtil
364 // FIXME: this is a temporary solution.
365 public static void prepareDialog(JDialog dialog) {
366 if (Main.pref.getBoolean("window-handling.option-pane-always-on-top", true)) {
367 try {
368 dialog.setAlwaysOnTop(true);
369 } catch(SecurityException e) {
370 System.out.println(tr("Warning: failed to put option pane dialog always on top. Exception was: {0}", e.toString()));
371 }
372 }
373 dialog.setModal(true);
374 dialog.toFront();
375 dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
376 }
377
378 /**
379 * Adds the WMSLayer following this rule:<br/>
380 * - if a WMSLayer exists place this new layer just before this layer<br/>
381 * - Otherwise place it at the bottom
382 * @param wmsLayer the wmsLayer to add
383 */
384 public static void addWMSLayer(WMSLayer wmsLayer) {
385 if (Main.map != null && Main.map.mapView != null) {
386 int wmsNewLayerPos = Main.map.mapView.getAllLayers().size();
387 for(Layer l : Main.map.mapView.getLayersOfType(WMSLayer.class)) {
388 int wmsPos = Main.map.mapView.getLayerPos(l);
389 if (wmsPos < wmsNewLayerPos) wmsNewLayerPos = wmsPos;
390 }
391 Main.main.addLayer(wmsLayer);
392 // Move the layer to its new position
393 Main.map.mapView.moveLayer(wmsLayer, wmsNewLayerPos);
394 } else
395 Main.main.addLayer(wmsLayer);
396 }
397
398 private static String checkSourceMillesime() {
399 java.util.Calendar calendar = java.util.Calendar.getInstance();
400 int currentYear = calendar.get(java.util.Calendar.YEAR);
401 String src = Main.pref.get("cadastrewms.source",
402 "cadastre-dgi-fr source : Direction G\u00e9n\u00e9rale des Imp\u00f4ts - Cadastre. Mise \u00e0 jour : AAAA");
403 String srcYear = src.substring(src.lastIndexOf(" ")+1);
404 Integer year = null;
405 try {
406 year = Integer.decode(srcYear);
407 } catch (NumberFormatException e) {}
408 if (srcYear.equals("AAAA") || (year != null && year < currentYear)) {
409 System.out.println("Replace source year "+srcYear+" by current year "+currentYear);
410 src = src.substring(0, src.lastIndexOf(" ")+1)+currentYear;
411 Main.pref.put("cadastrewms.source", src);
412 }
413 return src;
414 }
415
416}
Note: See TracBrowser for help on using the repository browser.