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

Last change on this file since 20484 was 20484, checked in by pieren, 15 years ago

minor fixes

  • Property svn:eol-style set to native
File size: 19.0 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 */
112public class CadastrePlugin extends Plugin {
113 static String VERSION = "1.8";
114
115 static JMenu cadastreJMenu;
116
117 public static CadastreGrabber cadastreGrabber = new CadastreGrabber();
118
119 public static String source = "";
120
121 // true if the checkbox "auto-sourcing" is set in the plugin menu
122 public static boolean autoSourcing = false;
123
124 // true when the plugin is first used, e.g. grab from WMS or download cache file
125 public static boolean pluginUsed = false;
126
127 public static String cacheDir = null;
128
129 public static boolean alterColors = false;
130
131 public static boolean backgroundTransparent = false;
132
133 public static float transparency = 1.0f;
134
135 public static boolean drawBoundaries = false;
136
137 public static int imageWidth, imageHeight;
138
139 public static String grabLayers, grabStyles = null;
140
141 static private boolean menuEnabled = false;
142
143 /**
144 * Creates the plugin and setup the default settings if necessary
145 *
146 * @throws Exception
147 */
148 public CadastrePlugin(PluginInformation info) throws Exception {
149 super(info);
150 System.out.println("Pluging cadastre-fr v"+VERSION+" started...");
151 if (Main.pref.get("cadastrewms.cacheDir").equals(""))
152 cacheDir = Main.pref.getPreferencesDir()+"plugins"+File.separatorChar+"cadastrewms"+File.separatorChar;
153 else {
154 cacheDir = Main.pref.get("cadastrewms.cacheDir");
155 if (cacheDir.charAt(cacheDir.length()-1) != File.separatorChar )
156 cacheDir += File.separatorChar;
157 }
158 System.out.println("current cache directory: "+cacheDir);
159
160 refreshConfiguration();
161
162 UploadAction.registerUploadHook(new CheckSourceUploadHook());
163
164 }
165
166 public static void refreshMenu() {
167 MainMenu menu = Main.main.menu;
168
169 if (cadastreJMenu == null) {
170 cadastreJMenu = menu.addMenu(marktr("Cadastre"), KeyEvent.VK_C, menu.defaultMenuPos, ht("/Plugin/CadastreFr"));
171 JosmAction grab = new MenuActionGrab();
172 JMenuItem menuGrab = new JMenuItem(grab);
173 KeyStroke ks = grab.getShortcut().getKeyStroke();
174 if (ks != null) {
175 menuGrab.setAccelerator(ks);
176 }
177 JMenuItem menuActionGrabPlanImage = new JMenuItem(new MenuActionGrabPlanImage());
178 JMenuItem menuSettings = new JMenuItem(new MenuActionNewLocation());
179 final JCheckBoxMenuItem menuSource = new JCheckBoxMenuItem(tr("Auto sourcing"));
180 menuSource.setSelected(autoSourcing);
181 menuSource.addActionListener(new ActionListener() {
182 public void actionPerformed(ActionEvent ev) {
183 Main.pref.put("cadastrewms.autosourcing", menuSource.isSelected());
184 autoSourcing = menuSource.isSelected();
185 }
186 });
187
188 //JMenuItem menuResetCookie = new JMenuItem(new MenuActionResetCookie());
189 //JMenuItem menuLambertZone = new JMenuItem(new MenuActionLambertZone());
190 JMenuItem menuLoadFromCache = new JMenuItem(new MenuActionLoadFromCache());
191 // temporary disabled:
192 //JMenuItem menuActionBoundaries = new JMenuItem(new MenuActionBoundaries());
193 JMenuItem menuActionBuildings = new JMenuItem(new MenuActionBuildings());
194
195 cadastreJMenu.add(menuGrab);
196 cadastreJMenu.add(menuActionGrabPlanImage);
197 cadastreJMenu.add(menuSettings);
198 cadastreJMenu.add(menuSource);
199 //cadastreJMenu.add(menuResetCookie); not required any more
200 //cadastreJMenu.add(menuLambertZone);
201 if (Main.pref.getBoolean("cadastrewms.buildingsMenu", false))
202 cadastreJMenu.add(menuActionBuildings);
203 cadastreJMenu.add(menuLoadFromCache);
204 // all SVG features disabled until official WMS is released
205 //cadastreJMenu.add(menuActionBoundaries);
206 }
207 setEnabledAll(menuEnabled);
208 }
209
210 public static void refreshConfiguration() {
211 source = checkSourceMillesime();
212 autoSourcing = Main.pref.getBoolean("cadastrewms.autosourcing", true);
213 alterColors = Main.pref.getBoolean("cadastrewms.alterColors");
214 drawBoundaries = Main.pref.getBoolean("cadastrewms.drawBoundaries", false);
215 if (alterColors) {
216 backgroundTransparent = Main.pref.getBoolean("cadastrewms.backgroundTransparent");
217 transparency = Float.parseFloat(Main.pref.get("cadastrewms.brightness", "1.0f"));
218 } else {
219 backgroundTransparent = false;
220 transparency = 1.0f;
221 }
222 String currentResolution = Main.pref.get("cadastrewms.resolution", "high");
223 if (currentResolution.equals("high")) {
224 imageWidth = 1000; imageHeight = 800;
225 } else if (currentResolution.equals("medium")){
226 imageWidth = 800; imageHeight = 600;
227 } else {
228 imageWidth = 600; imageHeight = 400;
229 }
230 refreshLayersURL();
231
232 // overwrite F11 shortcut used from the beginning by this plugin and recently used
233 // for full-screen switch in JOSM core
234 int i = 0;
235 String p = Main.pref.get("shortcut.shortcut."+i, null);
236 boolean alreadyRedefined = false;
237 while (p != null) {
238 String[] s = p.split(";");
239 alreadyRedefined = alreadyRedefined || s[0].equals("menu:view:fullscreen");
240 i++;
241 p = Main.pref.get("shortcut.shortcut."+i, null);
242 }
243 if (!alreadyRedefined) {
244 int reply = JOptionPane.showConfirmDialog(null,
245 tr("Plugin cadastre-fr used traditionaly for grabbing the key shortcut F11\n"+
246 "which is currently allocated for full-screen switch by default\n"+
247 "Would you like to restore F11 for grab action ?"),
248 tr("Restore grab shortcut F11"),
249 JOptionPane.YES_NO_OPTION);
250 if (reply == JOptionPane.OK_OPTION) {
251 System.out.println("redefine fullscreen shortcut F11 to shift+F11");
252 Main.pref.put("shortcut.shortcut."+i, "menu:view:fullscreen;Toggle Full Screen view;122;5;122;64;false;true");
253 JOptionPane.showMessageDialog(Main.parent,tr("JOSM is stopped for the change to take effect."));
254 System.exit(0);
255 }
256 } else
257 System.out.println("shortcut F11 already redefined; do not change");
258
259 refreshMenu();
260 }
261
262 private static void refreshLayersURL() {
263 grabLayers = "";
264 grabStyles = "";
265 if (Main.pref.getBoolean("cadastrewms.layerWater", true)) {
266 grabLayers += "CDIF:LS3,";
267 grabStyles += "LS3_90,";
268 }
269 if (Main.pref.getBoolean("cadastrewms.layerBuilding", true)) {
270 grabLayers += "CDIF:LS2,";
271 grabStyles += "LS2_90,";
272 }
273 if (Main.pref.getBoolean("cadastrewms.layerSymbol", true)) {
274 grabLayers += "CDIF:LS1,";
275 grabStyles += "LS1_90,";
276 }
277 if (Main.pref.getBoolean("cadastrewms.layerParcel", true)) {
278 grabLayers += "CDIF:PARCELLE,";
279 grabStyles += "PARCELLE_90,";
280 }
281 if (Main.pref.getBoolean("cadastrewms.layerNumero", true)) {
282 grabLayers += "CDIF:NUMERO,";
283 grabStyles += "NUMERO_90,";
284 }
285 if (Main.pref.getBoolean("cadastrewms.layerLabel", true)) {
286 grabLayers += "CDIF:PT3,CDIF:PT2,CDIF:PT1,";
287 grabStyles += "PT3_90,PT2_90,PT1_90,";
288 }
289 if (Main.pref.getBoolean("cadastrewms.layerLieudit", true)) {
290 grabLayers += "CDIF:LIEUDIT,";
291 grabStyles += "LIEUDIT_90,";
292 }
293 if (Main.pref.getBoolean("cadastrewms.layerSection", true)) {
294 grabLayers += "CDIF:SUBSECTION,CDIF:SECTION,";
295 grabStyles += "SUBSECTION_90,SECTION_90,";
296 }
297 if (Main.pref.getBoolean("cadastrewms.layerCommune", true)) {
298 grabLayers += "CDIF:COMMUNE,";
299 grabStyles += "COMMUNE_90,";
300 }
301 if (grabLayers.length() > 0) { // remove the last ','
302 grabLayers = grabLayers.substring(0, grabLayers.length()-1);
303 grabStyles = grabStyles.substring(0, grabStyles.length()-1);
304 }
305 }
306
307 @Override
308 public PreferenceSetting getPreferenceSetting() {
309 return new CadastrePreferenceSetting();
310 }
311
312 private static void setEnabledAll(boolean isEnabled) {
313 for (int i = 0; i < cadastreJMenu.getItemCount(); i++) {
314 JMenuItem item = cadastreJMenu.getItem(i);
315 if (item != null)
316 if (item.getText().equals(MenuActionGrabPlanImage.name) /*||
317 item.getText().equals(MenuActionGrab.name) ||
318 item.getText().equals(MenuActionBoundaries.name) ||
319 item.getText().equals(MenuActionBuildings.name)*/) {
320 item.setEnabled(isEnabled);
321 }
322 }
323 menuEnabled = isEnabled;
324 }
325
326 public void mapFrameInitialized(MapFrame oldFrame, MapFrame newFrame) {
327 if (cadastreJMenu != null) {
328 if (oldFrame == null && newFrame != null) {
329 setEnabledAll(true);
330 Main.map.addMapMode(new IconToggleButton(new WMSAdjustAction(Main.map)));
331 if (Main.pref.getBoolean("cadastrewms.buildingsMenu", false))
332 Main.map.addMapMode(new IconToggleButton(new Buildings(Main.map)));
333 } else if (oldFrame != null && newFrame == null) {
334 setEnabledAll(false);
335 //Lambert.layoutZone = -1;
336 //LambertCC9Zones.layoutZone = -1;
337 }
338 }
339 }
340
341 public static boolean isCadastreProjection() {
342 return Main.proj.toString().equals(new Lambert().toString())
343 || Main.proj.toString().equals(new UTM_20N_France_DOM().toString())
344 || Main.proj.toString().equals(new GaussLaborde_Reunion().toString())
345 || Main.proj.toString().equals(new LambertCC9Zones().toString());
346 }
347
348 public static void safeSleep(long milliseconds) {
349 try {
350 Thread.sleep(milliseconds);
351 } catch (InterruptedException e) {}
352 }
353
354 // See OptionPaneUtil
355 // FIXME: this is a temporary solution.
356 public static void prepareDialog(JDialog dialog) {
357 if (Main.pref.getBoolean("window-handling.option-pane-always-on-top", true)) {
358 try {
359 dialog.setAlwaysOnTop(true);
360 } catch(SecurityException e) {
361 System.out.println(tr("Warning: failed to put option pane dialog always on top. Exception was: {0}", e.toString()));
362 }
363 }
364 dialog.setModal(true);
365 dialog.toFront();
366 dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
367 }
368
369 /**
370 * Adds the WMSLayer following this rule:<br/>
371 * - if a WMSLayer exists place this new layer just before this layer<br/>
372 * - Otherwise place it at the bottom
373 * @param wmsLayer the wmsLayer to add
374 */
375 public static void addWMSLayer(WMSLayer wmsLayer) {
376 if (Main.map != null && Main.map.mapView != null) {
377 int wmsNewLayerPos = Main.map.mapView.getAllLayers().size();
378 for(Layer l : Main.map.mapView.getLayersOfType(WMSLayer.class)) {
379 int wmsPos = Main.map.mapView.getLayerPos(l);
380 if (wmsPos < wmsNewLayerPos) wmsNewLayerPos = wmsPos;
381 }
382 Main.main.addLayer(wmsLayer);
383 // Move the layer to its new position
384 Main.map.mapView.moveLayer(wmsLayer, wmsNewLayerPos);
385 } else
386 Main.main.addLayer(wmsLayer);
387 }
388
389 private static String checkSourceMillesime() {
390 java.util.Calendar calendar = java.util.Calendar.getInstance();
391 int currentYear = calendar.get(java.util.Calendar.YEAR);
392 String src = Main.pref.get("cadastrewms.source",
393 "cadastre-dgi-fr source : Direction G\u00e9n\u00e9rale des Imp\u00f4ts - Cadastre. Mise \u00e0 jour : AAAA");
394 String srcYear = src.substring(src.lastIndexOf(" ")+1);
395 Integer year = null;
396 try {
397 year = Integer.decode(srcYear);
398 } catch (NumberFormatException e) {}
399 if (srcYear.equals("AAAA") || (year != null && year < currentYear)) {
400 System.out.println("Replace source year "+srcYear+" by current year "+currentYear);
401 src = src.substring(0, src.lastIndexOf(" ")+1)+currentYear;
402 Main.pref.put("cadastrewms.source", src);
403 }
404 return src;
405 }
406
407}
Note: See TracBrowser for help on using the repository browser.