Changeset 29273 in osm for applications/editors/josm/plugins/CommandLine/src
- Timestamp:
- 2013-02-21T23:35:15+01:00 (12 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
applications/editors/josm/plugins/CommandLine/src/CommandLine/CommandLine.java
r28618 r29273 38 38 import java.util.ArrayList; 39 39 import java.util.Collection; 40 import java.util.HashMap;41 40 import java.util.List; 42 41 … … 66 65 import org.openstreetmap.josm.plugins.Plugin; 67 66 import org.openstreetmap.josm.plugins.PluginInformation; 67 import org.openstreetmap.josm.tools.SubclassFilteredCollection; 68 68 69 69 public class CommandLine extends Plugin { 70 protected JTextField textField; 71 protected JTextField historyField; 72 private String prefix; 73 private Mode mode; 74 private ArrayList<Command> commands; 75 private JMenu commandMenu; 76 protected Command currentCommand; 77 protected String commandSymbol; 78 protected History history; 79 protected MapFrame currentMapFrame; 80 protected MapMode previousMode; 81 82 static final String pluginDir = Main.pref.getPluginsDirectory().getAbsolutePath() + "/CommandLine/"; 83 84 public CommandLine(PluginInformation info) { 85 super(info); 86 commandSymbol = ": "; 87 history = new History(100); 88 historyField = new JTextField(); 89 textField = new JTextField() { 90 @Override 91 protected void processKeyEvent(KeyEvent e) { 92 if (e.getID() == KeyEvent.KEY_PRESSED) { 93 String text = textField.getText(); 94 int code = e.getKeyCode(); 95 if (code == KeyEvent.VK_ENTER) { 96 String commandText = textField.getText().substring(prefix.length()); 97 switch (mode) { 98 case IDLE: 99 if (commandText.isEmpty()) { 100 commandText = history.getLastItem(); 101 } 102 else { 103 history.addItem(commandText); 104 } 105 Command command = findCommand(commandText, true); 106 if (command != null) { 107 startCommand(command); 108 } 109 else 110 setMode(Mode.IDLE); 111 break; 112 case SELECTION: 113 if (currentMapFrame.mapMode instanceof WayAction || currentMapFrame.mapMode instanceof NodeAction || currentMapFrame.mapMode instanceof RelationAction || currentMapFrame.mapMode instanceof AnyAction) { 114 Collection<OsmPrimitive> selected = Main.main.getCurrentDataSet().getSelected(); 115 if (selected.size() > 0) 116 loadParameter(selected, true); 117 } 118 else { 119 loadParameter(commandText, currentCommand.parameters.get(currentCommand.currentParameterNum).maxInstances == 1); 120 } 121 break; 122 case ADJUSTMENT: 123 break; 124 } 125 e.consume(); 126 } 127 else if (code == KeyEvent.VK_UP) { 128 textField.setText(prefix + history.getPrevItem()); 129 e.consume(); 130 } 131 else if (code == KeyEvent.VK_DOWN) { 132 textField.setText(prefix + history.getNextItem()); 133 e.consume(); 134 } 135 else if (code == KeyEvent.VK_BACK_SPACE || code == KeyEvent.VK_LEFT) { 136 if (textField.getCaretPosition() <= prefix.length()) 137 e.consume(); 138 } 139 else if (code == KeyEvent.VK_HOME) { 140 setCaretPosition(prefix.length()); 141 e.consume(); 142 } 143 else if (code == KeyEvent.VK_ESCAPE) { 144 if (textField.getText().length() == prefix.length() && mode == Mode.IDLE) 145 deactivate(); 146 else 147 endInput(); 148 e.consume(); 149 } 150 else if (code == KeyEvent.VK_DELETE || code == KeyEvent.VK_RIGHT || code == KeyEvent.VK_END) { 151 } 152 else { 153 e.consume(); 154 } 155 if (textField.getCaretPosition() < prefix.length() || (textField.getSelectionStart() < prefix.length() && textField.getSelectionStart() > 0) ) 156 e.consume(); 157 } 158 if (e.getID() == KeyEvent.KEY_TYPED) 159 if (textField.getCaretPosition() < prefix.length() || (textField.getSelectionStart() < prefix.length() && textField.getSelectionStart() > 0) ) 160 e.consume(); 161 super.processKeyEvent(e); 162 if (textField.getText().length() < prefix.length()) { // Safe 163 setMode(mode); 164 } 165 if (e.getID() == KeyEvent.KEY_TYPED) { 166 if (e.getKeyChar() > 'A' && e.getKeyChar() < 'z') { 167 Command command = findCommand(textField.getText().substring(prefix.length()), false); 168 if (command != null) { 169 int currentPos = textField.getSelectionStart() == 0 ? textField.getCaretPosition() : textField.getSelectionStart(); 170 textField.setText(prefix + command.name); 171 textField.setCaretPosition(currentPos); 172 textField.select(currentPos, prefix.length() + command.name.length()); 173 } 174 } 175 } 176 } 177 @Override 178 protected void processMouseEvent(MouseEvent e) { 179 super.processMouseEvent(e); 180 if (e.getButton() == MouseEvent.BUTTON1 && e.getID() == MouseEvent.MOUSE_RELEASED) { 181 if (textField.getSelectionStart() > 0 && textField.getSelectionStart() < prefix.length()) 182 textField.setSelectionStart(prefix.length()); 183 else if (textField.getCaretPosition() < prefix.length()) 184 textField.setCaretPosition(prefix.length()); 185 } 186 } 187 }; 188 189 if ( Main.main.menu != null ) { 190 commandMenu = Main.main.menu.addMenu(marktr("Commands") , KeyEvent.VK_M, Main.main.menu.defaultMenuPos, ht("/Plugin/CommandLine")); 191 MainMenu.add(Main.main.menu.toolsMenu, new CommandLineAction(this)); 192 } 193 loadCommands(); 194 setMode(Mode.IDLE); 195 } 196 197 public void startCommand(String commandName) { 198 Command command = findCommand(commandName, true); 199 if (command != null) { 200 startCommand(command); 201 } 202 } 203 204 protected void startCommand(Command command) { 205 if (Main.map == null) 206 return; 207 DataSet ds = Main.main.getCurrentDataSet(); 208 if (ds == null) 209 return; 210 currentCommand = command; 211 currentCommand.resetLoading(); 212 parseSelection(ds.getSelected()); 213 if (!(Main.map.mapMode instanceof AnyAction || Main.map.mapMode instanceof DummyAction || Main.map.mapMode instanceof LengthAction || Main.map.mapMode instanceof NodeAction || Main.map.mapMode instanceof PointAction || Main.map.mapMode instanceof RelationAction || Main.map.mapMode instanceof WayAction)) { 214 previousMode = Main.map.mapMode; 215 } 216 if (currentCommand.currentParameterNum < currentCommand.parameters.size()) 217 setMode(Mode.SELECTION); 218 else 219 runTool(); 220 } 221 222 @Override 223 public void mapFrameInitialized(MapFrame oldFrame, MapFrame newFrame) 224 { 225 currentMapFrame = newFrame; 226 if (oldFrame == null && newFrame != null) { 227 JToolBar tb = new JToolBar(); 228 tb.setLayout(new BorderLayout()); 229 tb.setFloatable(false); 230 tb.setOrientation(JToolBar.HORIZONTAL); 231 tb.add(historyField, BorderLayout.NORTH); 232 tb.add(textField, BorderLayout.SOUTH); 233 currentMapFrame.add(tb, BorderLayout.NORTH); 234 printHistory("Loaded CommandLine, version " + getPluginInformation().version); 235 } 236 } 237 238 protected void printHistory(String text) { 239 historyField.setText(text); 240 } 241 242 private void loadCommands() { 243 commands = (new Loader(getPluginDir())).load(); 244 for (Command command : commands) { 245 commandMenu.add(new CommandAction(command, this)); 246 } 247 } 248 249 private Command findCommand(String text, boolean strict) { 250 for (int i = 0; i < commands.size(); i++) { 251 if (strict) { 252 if ( commands.get(i).name.equalsIgnoreCase(text) ) { 253 return commands.get(i); 254 } 255 } 256 else { 257 if ( commands.get(i).name.toLowerCase().startsWith( text.toLowerCase() ) && text.length() > 1 ) { 258 return commands.get(i); 259 } 260 } 261 } 262 return null; 263 } 264 265 protected void setMode(Mode targetMode) { 266 DataSet currentDataSet = Main.main.getCurrentDataSet(); 267 if (currentDataSet != null) { 268 currentDataSet.clearSelection(); 269 Main.map.mapView.repaint(); 270 } 271 if (targetMode == Mode.IDLE) { 272 mode = Mode.IDLE; 273 currentCommand = null; 274 prefix = tr("Command") + commandSymbol; 275 textField.setText(prefix); 276 } 277 else if (targetMode == Mode.SELECTION) { 278 mode = Mode.SELECTION; 279 Parameter currentParameter = currentCommand.parameters.get(currentCommand.currentParameterNum); 280 prefix = tr(currentParameter.description == null ? currentParameter.name : currentParameter.description); 281 if (currentParameter.getRawValue() instanceof Relay) 282 prefix = prefix + " (" + ((Relay)(currentParameter.getRawValue())).getOptionsString() + ")"; 283 prefix += commandSymbol; 284 String value = currentParameter.getValue(); 285 textField.setText(prefix + value); 286 Type currentType = currentParameter.type; 287 MapMode action = null; 288 switch (currentType) { 289 case POINT: 290 action = new PointAction(currentMapFrame, this); 291 break; 292 case WAY: 293 action = new WayAction(currentMapFrame, this); 294 break; 295 case NODE: 296 action = new NodeAction(currentMapFrame, this); 297 break; 298 case RELATION: 299 action = new RelationAction(currentMapFrame, this); 300 break; 301 case ANY: 302 action = new AnyAction(currentMapFrame, this); 303 break; 304 case LENGTH: 305 action = new LengthAction(currentMapFrame, this); 306 break; 307 case USERNAME: 308 loadParameter(Main.pref.get("osm-server.username", null), true); 309 action = new DummyAction(currentMapFrame, this); 310 break; 311 case IMAGERYURL: 312 Layer layer = Main.map.mapView.getActiveLayer(); 313 if (layer != null) { 314 if (layer instanceof ImageryLayer) { 315 } 316 else { 317 List<ImageryLayer> imageryLayers = Main.map.mapView.getLayersOfType(ImageryLayer.class); 318 if (imageryLayers.size() == 1) { 319 layer = imageryLayers.get(0); 320 } 321 else { 322 endInput(); 323 return; 324 } 325 } 326 } 327 ImageryInfo info = ((ImageryLayer)layer).getInfo(); 328 String url = info.getUrl(); 329 String itype = info.getImageryType().getUrlString(); 330 loadParameter((url.equals("") ? itype : url), true); 331 action = new DummyAction(currentMapFrame, this); 332 break; 333 case IMAGERYOFFSET: 334 Layer olayer = Main.map.mapView.getActiveLayer(); 335 if (olayer != null) { 336 if (olayer instanceof ImageryLayer) { 337 } 338 else { 339 List<ImageryLayer> imageryLayers = Main.map.mapView.getLayersOfType(ImageryLayer.class); 340 if (imageryLayers.size() == 1) { 341 olayer = imageryLayers.get(0); 342 } 343 else { 344 endInput(); 345 return; 346 } 347 } 348 } 349 loadParameter((String.valueOf(((ImageryLayer)olayer).getDx()) + "," + String.valueOf(((ImageryLayer)olayer).getDy())), true); 350 action = new DummyAction(currentMapFrame, this); 351 break; 352 default: 353 action = new DummyAction(currentMapFrame, this); 354 break; 355 } 356 currentMapFrame.selectMapMode(action); 357 activate(); 358 textField.select(prefix.length(), textField.getText().length()); 359 } 360 else if (targetMode == Mode.PROCESSING) { 361 mode = Mode.PROCESSING; 362 prefix = tr("Processing..."); 363 textField.setText(prefix); 364 Main.map.mapView.setCursor(Cursor.getPredefinedCursor(Cursor.WAIT_CURSOR)); 365 } 366 } 367 368 public void activate() { 369 textField.requestFocus(); 370 textField.setCaretPosition(textField.getText().length()); 371 } 372 373 public void deactivate() { 374 Main.map.mapView.requestFocus(); 375 } 376 377 public void abortInput() { 378 printHistory(tr("Aborted") + "."); 379 endInput(); 380 } 381 382 public void endInput() { 383 setMode(Mode.IDLE); 384 Main.map.selectMapMode(previousMode); 385 Main.map.mapView.repaint(); 386 } 387 388 public void loadParameter(Object obj, boolean next) { 389 if (currentCommand.loadObject(obj)) { 390 if (currentCommand.hasNextParameter()) { 391 if (next) { 392 Parameter currentParameter = currentCommand.parameters.get(currentCommand.currentParameterNum); 393 String prefix = tr(currentParameter.description == null ? currentParameter.name : currentParameter.description); 394 prefix += commandSymbol; 395 String value = currentParameter.getValue(); 396 printHistory(prefix + value); 397 currentCommand.nextParameter(); 398 setMode(Mode.SELECTION); 399 } 400 } 401 else { 402 runTool(); 403 } 404 } 405 else { 406 System.out.println("Invalid argument"); 407 endInput(); 408 } 409 } 410 411 private void parseSelection(Collection<OsmPrimitive> selection) { 412 boolean ok = false; 413 for (OsmPrimitive obj : selection) { 414 ok = currentCommand.loadObject(obj); 415 if (!ok) 416 break; 417 } 418 if (ok) { 419 currentCommand.nextParameter(); 420 } 421 else { 422 currentCommand.resetLoading(); 423 } 424 //System.out.println("Selected before " + String.valueOf(currentCommand.currentParameterNum) + "\n"); 425 } 426 427 private class ToolProcess { 428 public Process process; 429 public volatile boolean running; 430 } 431 432 // Thanks to Upliner 433 public void runTool() { 434 setMode(Mode.PROCESSING); 435 String commandToRun = currentCommand.run; 436 final boolean tracks = currentCommand.tracks; 437 final ArrayList<Parameter> parameters = currentCommand.parameters; 438 439 for (Parameter parameter : currentCommand.parameters) { 440 commandToRun = commandToRun.replace("{" + parameter.name + "}", parameter.getValue()); 441 } 442 for (Parameter parameter : currentCommand.optParameters) { 443 commandToRun = commandToRun.replace("{" + parameter.name + "}", parameter.getValue()); 444 } 445 String[] listToRun = commandToRun.split(" "); 446 447 // create the process 448 final Object syncObj = new Object(); 449 450 ProcessBuilder builder; 451 builder = new ProcessBuilder(listToRun); 452 builder.directory(new File(getPluginDir())); 453 454 final StringBuilder debugstr = new StringBuilder(); 455 456 // debug: print resulting cmdline 457 for (String s : builder.command()) 458 debugstr.append(s + " "); 459 debugstr.append("\n"); 460 System.out.print(debugstr.toString()); 461 462 final ToolProcess tp = new ToolProcess(); 463 try { 464 tp.process = builder.start(); 465 } catch (final IOException e) { 466 e.printStackTrace(); 467 synchronized (debugstr) { 468 System.out.print( 469 tr("Error executing the script: ") + 470 debugstr.toString() + e.getMessage() + "\n" + e.getStackTrace()); 471 } 472 return; 473 } 474 tp.running = true; 475 476 // redirect child process's stderr to JOSM stderr 477 new Thread(new Runnable() { 478 @Override 479 public void run() { 480 try { 481 byte[] buffer = new byte[1024]; 482 InputStream errStream = tp.process.getErrorStream(); 483 int len; 484 while ((len = errStream.read(buffer)) > 0) { 485 synchronized (debugstr) { 486 debugstr.append(new String(buffer, 0, len)); 487 } 488 System.err.write(buffer, 0, len); 489 } 490 } catch (IOException e) { 491 } 492 } 493 }).start(); 494 495 // Write stdin stream 496 Thread osmWriteThread = new Thread(new Runnable() { 497 @Override 498 public void run() { 499 BBox bbox = null; 500 final OutputStream outputStream = tp.process.getOutputStream(); 501 PrintWriter printWriter = null; 502 try { printWriter = new PrintWriter(new OutputStreamWriter(outputStream, "utf-8")); } 503 catch (Exception e) {e.printStackTrace();} 504 final OsmWriter osmWriter = OsmWriterFactory.createOsmWriter(printWriter, true, null); 505 Collection<OsmPrimitive> refObjects = currentCommand.getDepsObjects(); 506 Collection<OsmPrimitive> pObjects; 507 osmWriter.header(); 508 for (OsmPrimitive primitive : refObjects) { 509 if (primitive instanceof Node) 510 osmWriter.visit((Node)primitive); 511 else if (primitive instanceof Way) 512 osmWriter.visit((Way)primitive); 513 else if (primitive instanceof Relation) 514 osmWriter.visit((Relation)primitive); 515 if (bbox == null) 516 bbox = new BBox(primitive.getBBox()); 517 else 518 bbox.addPrimitive(primitive, 0.0); 519 } 520 osmWriter.footer(); 521 osmWriter.flush(); 522 for (Parameter parameter : parameters) { 523 if (!parameter.isOsm()) 524 continue; 525 osmWriter.header(); 526 pObjects = parameter.getParameterObjects(); 527 for (OsmPrimitive primitive : pObjects) { 528 if (primitive instanceof Node) 529 osmWriter.visit((Node)primitive); 530 else if (primitive instanceof Way) 531 osmWriter.visit((Way)primitive); 532 else if (primitive instanceof Relation) 533 osmWriter.visit((Relation)primitive); 534 if (bbox == null) 535 bbox = new BBox(primitive.getBBox()); 536 else 537 bbox.addPrimitive(primitive, 0.0); 538 } 539 osmWriter.footer(); 540 osmWriter.flush(); 541 } 542 if (tracks) { 543 final GpxWriter gpxWriter = new GpxWriter(printWriter); 544 GpxFilter gpxFilter = new GpxFilter(); 545 gpxFilter.initBboxFilter(bbox); 546 List<GpxLayer> gpxLayers = Main.map.mapView.getLayersOfType(GpxLayer.class); 547 for (GpxLayer gpxLayer : gpxLayers) { 548 gpxFilter.addGpxData(gpxLayer.data); 549 } 550 gpxWriter.write(gpxFilter.getGpxData()); 551 } 552 osmWriter.close(); 553 synchronized (syncObj) { 554 tp.running = false; 555 syncObj.notifyAll(); 556 } 557 } 558 559 }); 560 561 // Read stdout stream 562 final OsmToCmd osmToCmd = new OsmToCmd(this, Main.main.getCurrentDataSet()); 563 Thread osmParseThread = new Thread(new Runnable() { 564 @Override 565 public void run() { 566 try { 567 String commandName = currentCommand.name; 568 HashMap<Long, Long> inexiDMap = new HashMap<Long, Long>(); 569 final InputStream inputStream = tp.process.getInputStream(); 570 osmToCmd.parseStream(inputStream); 571 final List<org.openstreetmap.josm.command.Command> cmdlist = osmToCmd.getCommandList(); 572 if (!cmdlist.isEmpty()) { 573 SequenceCommand cmd = new SequenceCommand(commandName, cmdlist); 574 Main.main.undoRedo.add(cmd); 575 } 576 } 577 catch (Exception e) {} 578 finally { 579 synchronized (syncObj) { 580 tp.running = false; 581 syncObj.notifyAll(); 582 } 583 } 584 } 585 586 }); 587 588 osmParseThread.start(); 589 osmWriteThread.start(); 590 591 synchronized (syncObj) { 592 try { 593 syncObj.wait(10000); 594 } catch (InterruptedException e) { 595 } 596 } 597 if (tp.running) { 598 new Thread(new PleaseWaitRunnable(currentCommand.name) { 599 @Override 600 protected void realRun() { 601 try { 602 progressMonitor.indeterminateSubTask(null); 603 synchronized (syncObj) { 604 if (tp.running) 605 syncObj.wait(); 606 } 607 } catch (InterruptedException e) { 608 } 609 } 610 611 @Override 612 protected void cancel() { 613 synchronized (syncObj) { 614 tp.running = false; 615 tp.process.destroy(); 616 syncObj.notifyAll(); 617 endInput(); 618 } 619 } 620 621 @Override 622 protected void finish() { 623 } 624 }).start(); 625 } 626 endInput(); 627 } 70 protected JTextField textField; 71 protected JTextField historyField; 72 private String prefix; 73 private Mode mode; 74 private ArrayList<Command> commands; 75 private JMenu commandMenu; 76 protected Command currentCommand; 77 protected String commandSymbol; 78 protected History history; 79 protected MapFrame currentMapFrame; 80 protected MapMode previousMode; 81 82 static final String pluginDir = Main.pref.getPluginsDirectory().getAbsolutePath() + "/CommandLine/"; 83 84 @SuppressWarnings("serial") 85 public CommandLine(PluginInformation info) { 86 super(info); 87 commandSymbol = ": "; 88 history = new History(100); 89 historyField = new JTextField(); 90 textField = new JTextField() { 91 @Override 92 protected void processKeyEvent(KeyEvent e) { 93 if (e.getID() == KeyEvent.KEY_PRESSED) { 94 //String text = textField.getText(); 95 int code = e.getKeyCode(); 96 if (code == KeyEvent.VK_ENTER) { 97 String commandText = textField.getText().substring(prefix.length()); 98 switch (mode) { 99 case IDLE: 100 if (commandText.isEmpty()) { 101 commandText = history.getLastItem(); 102 } 103 else { 104 history.addItem(commandText); 105 } 106 Command command = findCommand(commandText, true); 107 if (command != null) { 108 startCommand(command); 109 } 110 else 111 setMode(Mode.IDLE); 112 break; 113 case SELECTION: 114 if (currentMapFrame.mapMode instanceof WayAction || currentMapFrame.mapMode instanceof NodeAction || currentMapFrame.mapMode instanceof RelationAction || currentMapFrame.mapMode instanceof AnyAction) { 115 Collection<OsmPrimitive> selected = Main.main.getCurrentDataSet().getSelected(); 116 if (selected.size() > 0) 117 loadParameter(selected, true); 118 } 119 else { 120 loadParameter(commandText, currentCommand.parameters.get(currentCommand.currentParameterNum).maxInstances == 1); 121 } 122 break; 123 case ADJUSTMENT: 124 break; 125 } 126 e.consume(); 127 } 128 else if (code == KeyEvent.VK_UP) { 129 textField.setText(prefix + history.getPrevItem()); 130 e.consume(); 131 } 132 else if (code == KeyEvent.VK_DOWN) { 133 textField.setText(prefix + history.getNextItem()); 134 e.consume(); 135 } 136 else if (code == KeyEvent.VK_BACK_SPACE || code == KeyEvent.VK_LEFT) { 137 if (textField.getCaretPosition() <= prefix.length()) 138 e.consume(); 139 } 140 else if (code == KeyEvent.VK_HOME) { 141 setCaretPosition(prefix.length()); 142 e.consume(); 143 } 144 else if (code == KeyEvent.VK_ESCAPE) { 145 if (textField.getText().length() == prefix.length() && mode == Mode.IDLE) 146 deactivate(); 147 else 148 endInput(); 149 e.consume(); 150 } 151 else if (code == KeyEvent.VK_DELETE || code == KeyEvent.VK_RIGHT || code == KeyEvent.VK_END) { 152 } 153 else { 154 e.consume(); 155 } 156 if (textField.getCaretPosition() < prefix.length() || (textField.getSelectionStart() < prefix.length() && textField.getSelectionStart() > 0) ) 157 e.consume(); 158 } 159 if (e.getID() == KeyEvent.KEY_TYPED) 160 if (textField.getCaretPosition() < prefix.length() || (textField.getSelectionStart() < prefix.length() && textField.getSelectionStart() > 0) ) 161 e.consume(); 162 super.processKeyEvent(e); 163 if (textField.getText().length() < prefix.length()) { // Safe 164 setMode(mode); 165 } 166 if (e.getID() == KeyEvent.KEY_TYPED) { 167 if (e.getKeyChar() > 'A' && e.getKeyChar() < 'z') { 168 Command command = findCommand(textField.getText().substring(prefix.length()), false); 169 if (command != null) { 170 int currentPos = textField.getSelectionStart() == 0 ? textField.getCaretPosition() : textField.getSelectionStart(); 171 textField.setText(prefix + command.name); 172 textField.setCaretPosition(currentPos); 173 textField.select(currentPos, prefix.length() + command.name.length()); 174 } 175 } 176 } 177 } 178 @Override 179 protected void processMouseEvent(MouseEvent e) { 180 super.processMouseEvent(e); 181 if (e.getButton() == MouseEvent.BUTTON1 && e.getID() == MouseEvent.MOUSE_RELEASED) { 182 if (textField.getSelectionStart() > 0 && textField.getSelectionStart() < prefix.length()) 183 textField.setSelectionStart(prefix.length()); 184 else if (textField.getCaretPosition() < prefix.length()) 185 textField.setCaretPosition(prefix.length()); 186 } 187 } 188 }; 189 190 if ( Main.main.menu != null ) { 191 commandMenu = Main.main.menu.addMenu(marktr("Commands") , KeyEvent.VK_M, Main.main.menu.defaultMenuPos, ht("/Plugin/CommandLine")); 192 MainMenu.add(Main.main.menu.toolsMenu, new CommandLineAction(this)); 193 } 194 loadCommands(); 195 setMode(Mode.IDLE); 196 } 197 198 public void startCommand(String commandName) { 199 Command command = findCommand(commandName, true); 200 if (command != null) { 201 startCommand(command); 202 } 203 } 204 205 protected void startCommand(Command command) { 206 if (Main.map == null) 207 return; 208 DataSet ds = Main.main.getCurrentDataSet(); 209 if (ds == null) 210 return; 211 currentCommand = command; 212 currentCommand.resetLoading(); 213 parseSelection(ds.getSelected()); 214 if (!(Main.map.mapMode instanceof AnyAction || Main.map.mapMode instanceof DummyAction || Main.map.mapMode instanceof LengthAction || Main.map.mapMode instanceof NodeAction || Main.map.mapMode instanceof PointAction || Main.map.mapMode instanceof RelationAction || Main.map.mapMode instanceof WayAction)) { 215 previousMode = Main.map.mapMode; 216 } 217 if (currentCommand.currentParameterNum < currentCommand.parameters.size()) 218 setMode(Mode.SELECTION); 219 else 220 runTool(); 221 } 222 223 @Override 224 public void mapFrameInitialized(MapFrame oldFrame, MapFrame newFrame) 225 { 226 currentMapFrame = newFrame; 227 if (oldFrame == null && newFrame != null) { 228 JToolBar tb = new JToolBar(); 229 tb.setLayout(new BorderLayout()); 230 tb.setFloatable(false); 231 tb.setOrientation(JToolBar.HORIZONTAL); 232 tb.add(historyField, BorderLayout.NORTH); 233 tb.add(textField, BorderLayout.SOUTH); 234 currentMapFrame.add(tb, BorderLayout.NORTH); 235 printHistory("Loaded CommandLine, version " + getPluginInformation().version); 236 } 237 } 238 239 protected void printHistory(String text) { 240 historyField.setText(text); 241 } 242 243 private void loadCommands() { 244 commands = (new Loader(getPluginDir())).load(); 245 for (Command command : commands) { 246 commandMenu.add(new CommandAction(command, this)); 247 } 248 } 249 250 private Command findCommand(String text, boolean strict) { 251 for (int i = 0; i < commands.size(); i++) { 252 if (strict) { 253 if ( commands.get(i).name.equalsIgnoreCase(text) ) { 254 return commands.get(i); 255 } 256 } 257 else { 258 if ( commands.get(i).name.toLowerCase().startsWith( text.toLowerCase() ) && text.length() > 1 ) { 259 return commands.get(i); 260 } 261 } 262 } 263 return null; 264 } 265 266 protected void setMode(Mode targetMode) { 267 DataSet currentDataSet = Main.main.getCurrentDataSet(); 268 if (currentDataSet != null) { 269 currentDataSet.clearSelection(); 270 Main.map.mapView.repaint(); 271 } 272 if (targetMode == Mode.IDLE) { 273 mode = Mode.IDLE; 274 currentCommand = null; 275 prefix = tr("Command") + commandSymbol; 276 textField.setText(prefix); 277 } 278 else if (targetMode == Mode.SELECTION) { 279 mode = Mode.SELECTION; 280 Parameter currentParameter = currentCommand.parameters.get(currentCommand.currentParameterNum); 281 prefix = tr(currentParameter.description == null ? currentParameter.name : currentParameter.description); 282 if (currentParameter.getRawValue() instanceof Relay) 283 prefix = prefix + " (" + ((Relay)(currentParameter.getRawValue())).getOptionsString() + ")"; 284 prefix += commandSymbol; 285 String value = currentParameter.getValue(); 286 textField.setText(prefix + value); 287 Type currentType = currentParameter.type; 288 MapMode action = null; 289 switch (currentType) { 290 case POINT: 291 action = new PointAction(currentMapFrame, this); 292 break; 293 case WAY: 294 action = new WayAction(currentMapFrame, this); 295 break; 296 case NODE: 297 action = new NodeAction(currentMapFrame, this); 298 break; 299 case RELATION: 300 action = new RelationAction(currentMapFrame, this); 301 break; 302 case ANY: 303 action = new AnyAction(currentMapFrame, this); 304 break; 305 case LENGTH: 306 action = new LengthAction(currentMapFrame, this); 307 break; 308 case USERNAME: 309 loadParameter(Main.pref.get("osm-server.username", null), true); 310 action = new DummyAction(currentMapFrame, this); 311 break; 312 case IMAGERYURL: 313 Layer layer = Main.map.mapView.getActiveLayer(); 314 if (layer != null) { 315 if (layer instanceof ImageryLayer) { 316 } 317 else { 318 List<ImageryLayer> imageryLayers = Main.map.mapView.getLayersOfType(ImageryLayer.class); 319 if (imageryLayers.size() == 1) { 320 layer = imageryLayers.get(0); 321 } 322 else { 323 endInput(); 324 return; 325 } 326 } 327 } 328 ImageryInfo info = ((ImageryLayer)layer).getInfo(); 329 String url = info.getUrl(); 330 String itype = info.getImageryType().getUrlString(); 331 loadParameter((url.equals("") ? itype : url), true); 332 action = new DummyAction(currentMapFrame, this); 333 break; 334 case IMAGERYOFFSET: 335 Layer olayer = Main.map.mapView.getActiveLayer(); 336 if (olayer != null) { 337 if (olayer instanceof ImageryLayer) { 338 } 339 else { 340 List<ImageryLayer> imageryLayers = Main.map.mapView.getLayersOfType(ImageryLayer.class); 341 if (imageryLayers.size() == 1) { 342 olayer = imageryLayers.get(0); 343 } 344 else { 345 endInput(); 346 return; 347 } 348 } 349 } 350 loadParameter((String.valueOf(((ImageryLayer)olayer).getDx()) + "," + String.valueOf(((ImageryLayer)olayer).getDy())), true); 351 action = new DummyAction(currentMapFrame, this); 352 break; 353 default: 354 action = new DummyAction(currentMapFrame, this); 355 break; 356 } 357 currentMapFrame.selectMapMode(action); 358 activate(); 359 textField.select(prefix.length(), textField.getText().length()); 360 } 361 else if (targetMode == Mode.PROCESSING) { 362 mode = Mode.PROCESSING; 363 prefix = tr("Processing..."); 364 textField.setText(prefix); 365 Main.map.mapView.setCursor(Cursor.getPredefinedCursor(Cursor.WAIT_CURSOR)); 366 } 367 } 368 369 public void activate() { 370 textField.requestFocus(); 371 textField.setCaretPosition(textField.getText().length()); 372 } 373 374 public void deactivate() { 375 Main.map.mapView.requestFocus(); 376 } 377 378 public void abortInput() { 379 printHistory(tr("Aborted") + "."); 380 endInput(); 381 } 382 383 public void endInput() { 384 setMode(Mode.IDLE); 385 Main.map.selectMapMode(previousMode); 386 Main.map.mapView.repaint(); 387 } 388 389 public void loadParameter(Object obj, boolean next) { 390 if (currentCommand.loadObject(obj)) { 391 if (currentCommand.hasNextParameter()) { 392 if (next) { 393 Parameter currentParameter = currentCommand.parameters.get(currentCommand.currentParameterNum); 394 String prefix = tr(currentParameter.description == null ? currentParameter.name : currentParameter.description); 395 prefix += commandSymbol; 396 String value = currentParameter.getValue(); 397 printHistory(prefix + value); 398 currentCommand.nextParameter(); 399 setMode(Mode.SELECTION); 400 } 401 } 402 else { 403 runTool(); 404 } 405 } 406 else { 407 System.out.println("Invalid argument"); 408 endInput(); 409 } 410 } 411 412 private void parseSelection(Collection<OsmPrimitive> selection) { 413 boolean ok = false; 414 for (OsmPrimitive obj : selection) { 415 ok = currentCommand.loadObject(obj); 416 if (!ok) 417 break; 418 } 419 if (ok) { 420 currentCommand.nextParameter(); 421 } 422 else { 423 currentCommand.resetLoading(); 424 } 425 //System.out.println("Selected before " + String.valueOf(currentCommand.currentParameterNum) + "\n"); 426 } 427 428 private class ToolProcess { 429 public Process process; 430 public volatile boolean running; 431 } 432 433 // Thanks to Upliner 434 public void runTool() { 435 setMode(Mode.PROCESSING); 436 String commandToRun = currentCommand.run; 437 final boolean tracks = currentCommand.tracks; 438 final ArrayList<Parameter> parameters = currentCommand.parameters; 439 440 for (Parameter parameter : currentCommand.parameters) { 441 commandToRun = commandToRun.replace("{" + parameter.name + "}", parameter.getValue()); 442 } 443 for (Parameter parameter : currentCommand.optParameters) { 444 commandToRun = commandToRun.replace("{" + parameter.name + "}", parameter.getValue()); 445 } 446 String[] listToRun = commandToRun.split(" "); 447 448 // create the process 449 final Object syncObj = new Object(); 450 451 ProcessBuilder builder; 452 builder = new ProcessBuilder(listToRun); 453 builder.directory(new File(getPluginDir())); 454 455 final StringBuilder debugstr = new StringBuilder(); 456 457 // debug: print resulting cmdline 458 for (String s : builder.command()) 459 debugstr.append(s + " "); 460 debugstr.append("\n"); 461 System.out.print(debugstr.toString()); 462 463 final ToolProcess tp = new ToolProcess(); 464 try { 465 tp.process = builder.start(); 466 } catch (final IOException e) { 467 e.printStackTrace(); 468 synchronized (debugstr) { 469 System.out.print( 470 tr("Error executing the script: ") + 471 debugstr.toString() + e.getMessage() + "\n" + e.getStackTrace()); 472 } 473 return; 474 } 475 tp.running = true; 476 477 // redirect child process's stderr to JOSM stderr 478 new Thread(new Runnable() { 479 @Override 480 public void run() { 481 try { 482 byte[] buffer = new byte[1024]; 483 InputStream errStream = tp.process.getErrorStream(); 484 int len; 485 while ((len = errStream.read(buffer)) > 0) { 486 synchronized (debugstr) { 487 debugstr.append(new String(buffer, 0, len)); 488 } 489 System.err.write(buffer, 0, len); 490 } 491 } catch (IOException e) { 492 } 493 } 494 }).start(); 495 496 // Write stdin stream 497 Thread osmWriteThread = new Thread(new Runnable() { 498 @Override 499 public void run() { 500 BBox bbox = null; 501 final OutputStream outputStream = tp.process.getOutputStream(); 502 PrintWriter printWriter = null; 503 try { printWriter = new PrintWriter(new OutputStreamWriter(outputStream, "utf-8")); } 504 catch (Exception e) {e.printStackTrace();} 505 final OsmWriter osmWriter = OsmWriterFactory.createOsmWriter(printWriter, true, null); 506 Collection<OsmPrimitive> refObjects = currentCommand.getDepsObjects(); 507 Collection<OsmPrimitive> pObjects; 508 osmWriter.header(); 509 Collection<OsmPrimitive> contents = new ArrayList<OsmPrimitive>(); 510 for (OsmPrimitive primitive : refObjects) { 511 contents.add(primitive); 512 if (bbox == null) 513 bbox = new BBox(primitive.getBBox()); 514 else 515 bbox.addPrimitive(primitive, 0.0); 516 } 517 for (Parameter parameter : parameters) { 518 if (!parameter.isOsm()) 519 continue; 520 pObjects = parameter.getParameterObjects(); 521 for (OsmPrimitive primitive : pObjects) { 522 contents.add(primitive); 523 if (bbox == null) 524 bbox = new BBox(primitive.getBBox()); 525 else 526 bbox.addPrimitive(primitive, 0.0); 527 } 528 } 529 osmWriter.writeNodes(new SubclassFilteredCollection<OsmPrimitive, Node>(contents, OsmPrimitive.nodePredicate)); 530 osmWriter.writeWays(new SubclassFilteredCollection<OsmPrimitive, Way>(contents, OsmPrimitive.wayPredicate)); 531 osmWriter.writeRelations(new SubclassFilteredCollection<OsmPrimitive, Relation>(contents, OsmPrimitive.relationPredicate)); 532 osmWriter.footer(); 533 osmWriter.flush(); 534 if (tracks) { 535 final GpxWriter gpxWriter = new GpxWriter(printWriter); 536 GpxFilter gpxFilter = new GpxFilter(); 537 gpxFilter.initBboxFilter(bbox); 538 List<GpxLayer> gpxLayers = Main.map.mapView.getLayersOfType(GpxLayer.class); 539 for (GpxLayer gpxLayer : gpxLayers) { 540 gpxFilter.addGpxData(gpxLayer.data); 541 } 542 gpxWriter.write(gpxFilter.getGpxData()); 543 } 544 osmWriter.close(); 545 synchronized (syncObj) { 546 tp.running = false; 547 syncObj.notifyAll(); 548 } 549 } 550 551 }); 552 553 // Read stdout stream 554 final OsmToCmd osmToCmd = new OsmToCmd(this, Main.main.getCurrentDataSet()); 555 Thread osmParseThread = new Thread(new Runnable() { 556 @Override 557 public void run() { 558 try { 559 String commandName = currentCommand.name; 560 //HashMap<Long, Long> inexiDMap = new HashMap<Long, Long>(); 561 final InputStream inputStream = tp.process.getInputStream(); 562 osmToCmd.parseStream(inputStream); 563 final List<org.openstreetmap.josm.command.Command> cmdlist = osmToCmd.getCommandList(); 564 if (!cmdlist.isEmpty()) { 565 SequenceCommand cmd = new SequenceCommand(commandName, cmdlist); 566 Main.main.undoRedo.add(cmd); 567 } 568 } 569 catch (Exception e) {} 570 finally { 571 synchronized (syncObj) { 572 tp.running = false; 573 syncObj.notifyAll(); 574 } 575 } 576 } 577 578 }); 579 580 osmParseThread.start(); 581 osmWriteThread.start(); 582 583 synchronized (syncObj) { 584 try { 585 syncObj.wait(10000); 586 } catch (InterruptedException e) { 587 } 588 } 589 if (tp.running) { 590 new Thread(new PleaseWaitRunnable(currentCommand.name) { 591 @Override 592 protected void realRun() { 593 try { 594 progressMonitor.indeterminateSubTask(null); 595 synchronized (syncObj) { 596 if (tp.running) 597 syncObj.wait(); 598 } 599 } catch (InterruptedException e) { 600 } 601 } 602 603 @Override 604 protected void cancel() { 605 synchronized (syncObj) { 606 tp.running = false; 607 tp.process.destroy(); 608 syncObj.notifyAll(); 609 endInput(); 610 } 611 } 612 613 @Override 614 protected void finish() { 615 } 616 }).start(); 617 } 618 endInput(); 619 } 628 620 }
Note:
See TracChangeset
for help on using the changeset viewer.