Ignore:
Timestamp:
2017-08-27T22:22:31+02:00 (7 years ago)
Author:
donvip
Message:

update to JOSM 12678

File:
1 edited

Legend:

Unmodified
Added
Removed
  • applications/editors/josm/plugins/OSMRecPlugin/src/org/openstreetmap/josm/plugins/osmrec/OSMRecPluginHelper.java

    r33525 r33579  
    107107//import org.openstreetmap.josm.gui.tagging.presets.TaggingPreset;
    108108import org.openstreetmap.josm.gui.util.GuiHelper;
     109import org.openstreetmap.josm.gui.util.WindowGeometry;
    109110import org.openstreetmap.josm.gui.widgets.PopupMenuLauncher;
    110111import org.openstreetmap.josm.io.XmlWriter;
     
    125126import org.openstreetmap.josm.tools.Logging;
    126127import org.openstreetmap.josm.tools.Shortcut;
    127 import org.openstreetmap.josm.tools.WindowGeometry;
    128128
    129129import com.vividsolutions.jts.geom.Coordinate;
     
    146146class OSMRecPluginHelper {
    147147
    148         private final DefaultTableModel tagData;
    149         private static Collection<String> fileHistory;
    150         private Map<File, Double> filesAndWeights = new HashMap<>();
    151         private boolean useCombinedModel;
    152 
    153         //the most recent file from history.
    154         //all necessary files will reside in a directory of this file
    155         private static String MAIN_PATH;
    156         private static String MODEL_PATH;
    157         private static String TEXTUAL_LIST_PATH;
    158         private static Map<String, List<String>> indirectClasses;
    159         private static Map<String, Integer> indirectClassesWithIDs;
    160         private static LanguageDetector languageDetector;
    161         private static String bestModelPath;
    162         private boolean modelWithClasses;
    163         private final String modelWithClassesPath;
    164         private boolean useCustomSVMModel = false;
    165         private String customSVMModelPath;
    166         private final String combinedModelClasses;
    167         private final String combinedModel;
    168 
    169         // Selection that we are editing by using both dialogs
    170         Collection<OsmPrimitive> sel;
    171 
    172         private String changedKey;
    173 
    174         Comparator<AutoCompletionListItem> defaultACItemComparator = new Comparator<AutoCompletionListItem>() {
    175                 @Override
    176                 public int compare(AutoCompletionListItem o1, AutoCompletionListItem o2) {
    177                         return String.CASE_INSENSITIVE_ORDER.compare(o1.getValue(), o2.getValue());
    178                 }
    179         };
    180 
    181         private String lastAddKey = null;
    182         private String lastAddValue = null;
    183 
    184         public static final int DEFAULT_LRU_TAGS_NUMBER = 5;
    185         public static final int MAX_LRU_TAGS_NUMBER = 30;
    186 
    187         // LRU cache for recently added tags (http://java-planet.blogspot.com/2005/08/how-to-set-up-simple-lru-cache-using.html)
    188         private final Map<Tag, Void> recentTags = new LinkedHashMap<Tag, Void>(MAX_LRU_TAGS_NUMBER+1, 1.1f, true) {
    189                 @Override
    190                 protected boolean removeEldestEntry(Map.Entry<Tag, Void> eldest) {
    191                         return size() > MAX_LRU_TAGS_NUMBER;
    192                 }
    193         };
    194 
    195         OSMRecPluginHelper(DefaultTableModel propertyData, Map<String, Map<String, Integer>> valueCount) {
    196                 this.tagData = propertyData;
    197                 fileHistory = Main.pref.getCollection("file-open.history");
    198                 if (!fileHistory.isEmpty()) {
    199                         MAIN_PATH = (String) fileHistory.toArray()[0];
    200                 } else {
    201                         MAIN_PATH = System.getProperty("user.home");
    202                 }
    203                 MODEL_PATH = new File(MAIN_PATH).getParentFile() + "/OSMRec_models";
    204                 TEXTUAL_LIST_PATH = MODEL_PATH + "/textualList.txt";
    205                 combinedModelClasses = MODEL_PATH + "/combinedModel.1";
    206                 combinedModel = MODEL_PATH + "/combinedModel.0";
    207                 bestModelPath = MODEL_PATH + "/best_model";
    208                 customSVMModelPath = bestModelPath;
    209                 modelWithClassesPath = MODEL_PATH + "/model_with_classes";
    210                 languageDetector = LanguageDetector.getInstance(MODEL_PATH + "/profiles");
    211 
    212                 SampleModelsExtractor sampleModelsExtractor = new SampleModelsExtractor();
    213 
    214                 sampleModelsExtractor.extractSampleSVMmodel("best_model", bestModelPath);
    215                 sampleModelsExtractor.extractSampleSVMmodel("model_with_classes", modelWithClassesPath);
    216         }
    217 
    218         /**
    219         * Open the add selection dialog and add a new key/value to the table (and to the dataset, of course).
    220         */
    221         public void addTag() {
    222                 changedKey = null;
    223                 sel = Main.main.getInProgressSelection();
    224                 if (sel == null || sel.isEmpty()) return;
    225 
    226                 final AddTagsDialog addDialog = new AddTagsDialog();
    227 
    228                 addDialog.showDialog();
    229 
    230                 addDialog.destroyActions();
    231                 if (addDialog.getValue() == 1)
    232                         addDialog.performTagAdding();
    233                 else
    234                         addDialog.undoAllTagsAdding();
    235         }
    236 
    237         /**
    238         * Edit the value in the tags table row.
    239         * @param row The row of the table from which the value is edited.
    240         * @param focusOnKey Determines if the initial focus should be set on key instead of value
    241         * @since 5653
    242         */
    243         public void editTag(final int row, boolean focusOnKey) {
    244                 changedKey = null;
    245                 sel = Main.main.getInProgressSelection();
    246                 String key = "";
    247 
    248                 @SuppressWarnings("unchecked")
    249                 Map<String, Integer> dumPar = new HashMap<>();
    250                 dumPar.put(" ", -1);
    251                 final TrainingDialog editDialog = new TrainingDialog(key, row,
    252                                 dumPar, focusOnKey);
    253                 editDialog.showDialog();
    254         }
    255 
    256         /**
    257         * If during last editProperty call user changed the key name, this key will be returned
    258         * Elsewhere, returns null.
    259         * @return The modified key, or {@code null}
    260         */
    261         public String getChangedKey() {
    262                 return changedKey;
    263         }
    264 
    265         /**
    266         * For a given key k, return a list of keys which are used as keys for
    267         * auto-completing values to increase the search space.
    268         * @param key the key k
    269         * @return a list of keys
    270         */
    271         private static List<String> getAutocompletionKeys(String key) {
    272                 if ("name".equals(key) || "addr:street".equals(key))
    273                         return Arrays.asList("addr:street", "name");
    274                 else
    275                         return Arrays.asList(key);
    276         }
    277 
    278         /**
    279         * Load recently used tags from preferences if needed.
    280         */
    281         public void loadTagsIfNeeded() {
    282                 if (PROPERTY_REMEMBER_TAGS.get() && recentTags.isEmpty()) {
    283                         recentTags.clear();
    284                         Collection<String> c = Main.pref.getCollection("properties.recent-tags");
    285                         Iterator<String> it = c.iterator();
    286                         String key, value;
    287                         while (it.hasNext()) {
    288                                 key = it.next();
    289                                 value = it.next();
    290                                 recentTags.put(new Tag(key, value), null);
    291                         }
    292                 }
    293         }
    294 
    295         /**
    296         * Store recently used tags in preferences if needed.
    297         */
    298         public void saveTagsIfNeeded() {
    299                 if (PROPERTY_REMEMBER_TAGS.get() && !recentTags.isEmpty()) {
    300                         List<String> c = new ArrayList<>(recentTags.size()*2);
    301                         for (Tag t: recentTags.keySet()) {
    302                                 c.add(t.getKey());
    303                                 c.add(t.getValue());
    304                         }
    305                         Main.pref.putCollection("properties.recent-tags", c);
    306                 }
    307         }
    308 
    309         /**
    310         * Warns user about a key being overwritten.
    311         * @param action The action done by the user. Must state what key is changed
    312         * @param togglePref  The preference to save the checkbox state to
    313         * @return {@code true} if the user accepts to overwrite key, {@code false} otherwise
    314         */
    315         private boolean warnOverwriteKey(String action, String togglePref) {
    316                 ExtendedDialog ed = new ExtendedDialog(
    317                                 Main.parent,
    318                                 tr("Overwrite key"),
    319                                 new String[]{tr("Replace"), tr("Cancel")});
    320                 ed.setButtonIcons(new String[]{"purge", "cancel"});
    321                 ed.setContent(action+"\n"+ tr("The new key is already used, overwrite values?"));
    322                 ed.setCancelButton(2);
    323                 ed.toggleEnable(togglePref);
    324                 ed.showDialog();
    325 
    326                 return ed.getValue() == 1;
    327         }
    328 
    329         public final class TrainingDialog extends AbstractTagsDialog {
    330 
    331                 private static final int FIELD_COLUMNS = 4;
    332                 private final JTextField inputFileField;
    333                 private final JLabel inputFileLabel;
    334                 private final JTextField topKField;
    335                 private final JTextField cParameterField;
    336                 private final JTextField frequencyField;
    337 
    338                 private final JButton fileBrowseButton;
    339                 private final JButton acceptConfigButton;
    340                 private JRadioButton frequencyButton;
    341                 private JRadioButton topKButton;
    342                 private JCheckBox cParameterCheckBox;
    343                 private final JButton resetConfigButton;
    344                 private String inputFileValue;
    345                 private Double cParameterValue = 0.0;
    346                 private Integer topKvalue = 0;
    347                 private Integer frequencyValue = 0;
    348                 private boolean crossValidateFlag;
    349                 private final JButton startTrainingButton;
    350                 private final JLabel cErrorMessageLabel;
    351                 private final JLabel topKErrorMessageLabel;
    352                 private final JLabel inputFileErrorMessageLabel;
    353                 private final JLabel frequencyErrorMessageLabel;
    354                 private final JProgressBar trainingProgressBar;
    355                 private final JRadioButton byAreaRadioButton;
    356                 private final JRadioButton byTimeRadioButton;
    357                 private final JLabel userNameLabel;
    358                 private final JTextField userNameField;
    359                 private final JTextField daysField;
    360                 private final JLabel daysLabel;
    361                 private final JCheckBox trainFromUserCheckBox;
    362                 private final JPanel userHistoryPanel;
    363                 private Integer daysValue;
    364                 private String usernameValue;
    365                 private TrainWorker trainWorker;
    366                 private UserDataExtractAndTrainWorker userDataExtractAndTrainWorker;
    367 
    368                 private TrainingDialog(String key, int row, Map<String, Integer> map, final boolean initialFocusOnKey) {
    369                         super(Main.parent, tr("Training process configuration"), new String[] {tr("Cancel")});
    370 
    371                         setButtonIcons(new String[] {"ok", "cancel"});
    372                         setCancelButton(2);
    373 
    374                         JPanel mainPanel = new JPanel(new BorderLayout(10, 10));   //6,6
    375                         JPanel configPanel = new JPanel(new BorderLayout(10, 10));  //6,6    //at NORTH of mainPanel
    376                         JPanel inputPanel = new JPanel(new FlowLayout(FlowLayout.LEFT));    //NORTH at config panel
    377                         JPanel paramPanel = new JPanel(new FlowLayout(FlowLayout.LEFT));    //WEST at config panel //config panel has EAST free
    378 
    379                         JPanel southPanel = new JPanel(new FlowLayout(FlowLayout.LEFT));    //SOUTH at config panel
    380                         userHistoryPanel = new JPanel(); //CENTER of config
    381 
    382                         trainingProgressBar = new JProgressBar(0, 100);
    383 
    384                         ButtonGroup paramGroup = new ButtonGroup();
    385                         ButtonGroup userGroup = new ButtonGroup();
    386                         inputFileLabel = new JLabel();
    387                         inputFileField = new JTextField();
    388                         cParameterField = new JTextField();
    389                         topKField = new JTextField();
    390                         frequencyField = new JTextField();
    391 
    392                         cParameterCheckBox = new JCheckBox("Define C parameter");
    393                         topKButton = new JRadioButton("Top-K terms");
    394                         frequencyButton = new JRadioButton("Max-Frequency");
    395                         fileBrowseButton = new JButton();
    396                         acceptConfigButton = new JButton("Accept Configuration");
    397                         resetConfigButton = new JButton("Reset Configuration/Cancel training");
    398                         startTrainingButton = new JButton("Train Model");
    399 
    400                         inputFileErrorMessageLabel = new JLabel("");
    401                         cErrorMessageLabel = new JLabel("");
    402                         topKErrorMessageLabel = new JLabel("");
    403                         frequencyErrorMessageLabel = new JLabel("");
    404 
    405                         trainFromUserCheckBox = new JCheckBox("Train Model From User");
    406                         byAreaRadioButton = new JRadioButton("By Area");
    407                         byTimeRadioButton = new JRadioButton("By Time");
    408                         userNameLabel = new JLabel("Username:");
    409                         userNameField = new JTextField();
    410 
    411                         daysLabel = new JLabel("Days: ");
    412                         daysField = new JTextField();
    413 
    414                         cParameterCheckBox.setSelected(true);
    415                         userHistoryPanel.setEnabled(false);
    416                         byAreaRadioButton.setEnabled(false);
    417                         byAreaRadioButton.setSelected(true);
    418                         byTimeRadioButton.setEnabled(false);
    419                         userNameLabel.setEnabled(false);
    420                         userNameField.setEnabled(false);
    421                         daysLabel.setEnabled(false);
    422                         daysField.setEnabled(false);
    423                         userNameField.setColumns(FIELD_COLUMNS);
    424                         daysField.setColumns(FIELD_COLUMNS);
    425 
    426                         Collection<String> fileHistory = Main.pref.getCollection("file-open.history");
    427                         if (!fileHistory.isEmpty()) {
    428                                 inputFileField.setText(MAIN_PATH);
    429                         }
    430 
    431                         fileBrowseButton.setText("...");
    432                         inputFileLabel.setText("OSM filepath: ");
    433                         inputFileErrorMessageLabel.setForeground(Color.RED);
    434                         inputFileErrorMessageLabel.setText("");
    435                         topKField.setText("50");
    436                         frequencyField.setText("200");
    437                         cParameterField.setText("0.01");
    438 
    439                         cParameterField.setColumns(FIELD_COLUMNS);
    440                         cParameterField.setEditable(false);
    441                         cParameterField.setHorizontalAlignment(javax.swing.SwingConstants.RIGHT);
    442                         cErrorMessageLabel.setForeground(Color.RED);
    443                         cErrorMessageLabel.setMinimumSize(new Dimension(150, 10));
    444 
    445                         topKButton.setSelected(true);
    446                         topKField.setColumns(FIELD_COLUMNS);
    447                         topKField.setHorizontalAlignment(javax.swing.SwingConstants.RIGHT);
    448                         topKErrorMessageLabel.setForeground(Color.RED);
    449 
    450                         frequencyField.setEditable(false);
    451                         frequencyField.setColumns(FIELD_COLUMNS);
    452                         frequencyField.setHorizontalAlignment(javax.swing.SwingConstants.RIGHT);
    453                         frequencyErrorMessageLabel.setForeground(Color.RED);
    454 
    455                         fileBrowseButton.addActionListener(new java.awt.event.ActionListener() {
    456                                 @Override
    457                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    458                                         inputFileChooserButtonActionPerformed(evt);
    459                                 }
    460                         });
    461                         topKButton.addActionListener(new java.awt.event.ActionListener() {
    462                                 @Override
    463                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    464                                         if (topKButton.isSelected()) {
    465                                                 frequencyField.setEditable(false);
    466                                                 topKField.setEditable(true);
    467                                         } else {
    468                                                 frequencyField.setEditable(true);
    469                                                 topKField.setEditable(false);
    470                                         }
    471                                 }
    472                         });
    473                         frequencyButton.addActionListener(new java.awt.event.ActionListener() {
    474                                 @Override
    475                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    476                                         if (frequencyButton.isSelected()) {
    477                                                 topKField.setEditable(false);
    478                                                 frequencyField.setEditable(true);
    479                                         } else {
    480                                                 topKField.setEditable(true);
    481                                                 frequencyField.setEditable(false);
    482                                         }
    483                                 }
    484                         });
    485                         cParameterCheckBox.addActionListener(new java.awt.event.ActionListener() {
    486                                 @Override
    487                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    488                                         if (cParameterCheckBox.isSelected()) {
    489                                                 cParameterField.setEditable(true);
    490                                         } else {
    491                                                 cParameterField.setEditable(false);
    492                                         }
    493                                 }
    494                         });
    495                         acceptConfigButton.addActionListener(new java.awt.event.ActionListener() {
    496                                 @Override
    497                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    498                                         acceptConfigButtonActionPerformed(evt);
    499                                 }
    500                         });
    501                         resetConfigButton.addActionListener(new java.awt.event.ActionListener() {
    502                                 @Override
    503                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    504                                         resetConfigButtonActionPerformed();
    505                                 }
    506                         });
    507 
    508                         startTrainingButton.addActionListener(new java.awt.event.ActionListener() {
    509                                 @Override
    510                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    511                                         if (!acceptConfigButton.isEnabled()) {
    512                                                 startTraining();
    513                                         } else {
    514                                                 System.out.println("Set values first!");
    515                                         }
    516                                 }
    517                         });
    518 
    519                         trainFromUserCheckBox.addActionListener(new java.awt.event.ActionListener() {
    520                                 @Override
    521                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    522                                         if (trainFromUserCheckBox.isSelected()) {
    523                                                 userHistoryPanel.setEnabled(true);
    524                                                 byAreaRadioButton.setEnabled(true);
    525                                                 byTimeRadioButton.setEnabled(true);
    526                                                 userNameLabel.setEnabled(true);
    527                                                 userNameField.setEnabled(true);
    528                                                 daysLabel.setEnabled(true);
    529                                                 daysField.setEnabled(true);
    530                                         } else {
    531                                                 userHistoryPanel.setEnabled(false);
    532                                                 byAreaRadioButton.setEnabled(false);
    533                                                 byTimeRadioButton.setEnabled(false);
    534                                                 userNameLabel.setEnabled(false);
    535                                                 userNameField.setEnabled(false);
    536                                                 daysLabel.setEnabled(false);
    537                                                 daysField.setEnabled(false);
    538                                         }
    539                                 }
    540                         });
    541 
    542                         byAreaRadioButton.addActionListener(new java.awt.event.ActionListener() {
    543                                 @Override
    544                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    545                                         if (byAreaRadioButton.isSelected()) {
    546                                                 daysField.setEditable(false);
    547                                         } else {
    548                                                 daysField.setEditable(true);
    549                                         }
    550                                 }
    551                         });
    552 
    553                         byTimeRadioButton.addActionListener(new java.awt.event.ActionListener() {
    554                                 @Override
    555                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    556                                         if (byTimeRadioButton.isSelected()) {
    557                                                 daysField.setEditable(true);
    558                                         } else {
    559                                                 daysField.setEditable(false);
    560                                         }
    561                                 }
    562                         });
    563 
    564                         //grouplayout for input panel
    565                         buildInputPanelGroupLayout(inputPanel);
    566 
    567                         //grouplayout for param panel
    568                         buildParamPanelGroupLayout(paramPanel);
    569 
    570                         inputPanel.add(inputFileLabel);
    571                         inputPanel.add(inputFileField);
    572                         inputPanel.add(fileBrowseButton);
    573                         inputPanel.add(inputFileErrorMessageLabel);
    574 
    575                         paramGroup.add(topKButton);
    576                         paramGroup.add(frequencyButton);
    577 
    578                         paramPanel.add(cParameterCheckBox);
    579                         paramPanel.add(cParameterField);
    580                         paramPanel.add(cErrorMessageLabel);
    581                         paramPanel.add(topKButton);
    582                         paramPanel.add(topKField);
    583                         paramPanel.add(topKErrorMessageLabel);
    584                         paramPanel.add(frequencyButton);
    585                         paramPanel.add(frequencyField);
    586                         paramPanel.add(frequencyErrorMessageLabel);
    587 
    588                         southPanel.add(acceptConfigButton);
    589                         southPanel.add(resetConfigButton);
    590                         southPanel.add(trainFromUserCheckBox);
    591 
    592                         userGroup.add(byAreaRadioButton);
    593                         userGroup.add(byTimeRadioButton);
    594                         userHistoryPanel.add(byAreaRadioButton);
    595                         userHistoryPanel.add(byTimeRadioButton);
    596                         userHistoryPanel.add(daysLabel);
    597                         userHistoryPanel.add(daysField);
    598                         userHistoryPanel.add(userNameLabel);
    599                         userHistoryPanel.add(userNameField);
    600 
    601                         //grouplayout for user history panel
    602                         /*
     148    private final DefaultTableModel tagData;
     149    private static Collection<String> fileHistory;
     150    private Map<File, Double> filesAndWeights = new HashMap<>();
     151    private boolean useCombinedModel;
     152
     153    //the most recent file from history.
     154    //all necessary files will reside in a directory of this file
     155    private static String MAIN_PATH;
     156    private static String MODEL_PATH;
     157    private static String TEXTUAL_LIST_PATH;
     158    private static Map<String, List<String>> indirectClasses;
     159    private static Map<String, Integer> indirectClassesWithIDs;
     160    private static LanguageDetector languageDetector;
     161    private static String bestModelPath;
     162    private boolean modelWithClasses;
     163    private final String modelWithClassesPath;
     164    private boolean useCustomSVMModel = false;
     165    private String customSVMModelPath;
     166    private final String combinedModelClasses;
     167    private final String combinedModel;
     168
     169    // Selection that we are editing by using both dialogs
     170    Collection<OsmPrimitive> sel;
     171
     172    private String changedKey;
     173
     174    Comparator<AutoCompletionListItem> defaultACItemComparator = new Comparator<AutoCompletionListItem>() {
     175        @Override
     176        public int compare(AutoCompletionListItem o1, AutoCompletionListItem o2) {
     177            return String.CASE_INSENSITIVE_ORDER.compare(o1.getValue(), o2.getValue());
     178        }
     179    };
     180
     181    private String lastAddKey = null;
     182    private String lastAddValue = null;
     183
     184    public static final int DEFAULT_LRU_TAGS_NUMBER = 5;
     185    public static final int MAX_LRU_TAGS_NUMBER = 30;
     186
     187    // LRU cache for recently added tags (http://java-planet.blogspot.com/2005/08/how-to-set-up-simple-lru-cache-using.html)
     188    private final Map<Tag, Void> recentTags = new LinkedHashMap<Tag, Void>(MAX_LRU_TAGS_NUMBER+1, 1.1f, true) {
     189        @Override
     190        protected boolean removeEldestEntry(Map.Entry<Tag, Void> eldest) {
     191            return size() > MAX_LRU_TAGS_NUMBER;
     192        }
     193    };
     194
     195    OSMRecPluginHelper(DefaultTableModel propertyData, Map<String, Map<String, Integer>> valueCount) {
     196        this.tagData = propertyData;
     197        fileHistory = Main.pref.getCollection("file-open.history");
     198        if (!fileHistory.isEmpty()) {
     199            MAIN_PATH = (String) fileHistory.toArray()[0];
     200        } else {
     201            MAIN_PATH = System.getProperty("user.home");
     202        }
     203        MODEL_PATH = new File(MAIN_PATH).getParentFile() + "/OSMRec_models";
     204        TEXTUAL_LIST_PATH = MODEL_PATH + "/textualList.txt";
     205        combinedModelClasses = MODEL_PATH + "/combinedModel.1";
     206        combinedModel = MODEL_PATH + "/combinedModel.0";
     207        bestModelPath = MODEL_PATH + "/best_model";
     208        customSVMModelPath = bestModelPath;
     209        modelWithClassesPath = MODEL_PATH + "/model_with_classes";
     210        languageDetector = LanguageDetector.getInstance(MODEL_PATH + "/profiles");
     211
     212        SampleModelsExtractor sampleModelsExtractor = new SampleModelsExtractor();
     213
     214        sampleModelsExtractor.extractSampleSVMmodel("best_model", bestModelPath);
     215        sampleModelsExtractor.extractSampleSVMmodel("model_with_classes", modelWithClassesPath);
     216    }
     217
     218    /**
     219    * Open the add selection dialog and add a new key/value to the table (and to the dataset, of course).
     220    */
     221    public void addTag() {
     222        changedKey = null;
     223        sel = Main.main.getInProgressSelection();
     224        if (sel == null || sel.isEmpty()) return;
     225
     226        final AddTagsDialog addDialog = new AddTagsDialog();
     227
     228        addDialog.showDialog();
     229
     230        addDialog.destroyActions();
     231        if (addDialog.getValue() == 1)
     232            addDialog.performTagAdding();
     233        else
     234            addDialog.undoAllTagsAdding();
     235    }
     236
     237    /**
     238    * Edit the value in the tags table row.
     239    * @param row The row of the table from which the value is edited.
     240    * @param focusOnKey Determines if the initial focus should be set on key instead of value
     241    * @since 5653
     242    */
     243    public void editTag(final int row, boolean focusOnKey) {
     244        changedKey = null;
     245        sel = Main.main.getInProgressSelection();
     246        String key = "";
     247
     248        @SuppressWarnings("unchecked")
     249        Map<String, Integer> dumPar = new HashMap<>();
     250        dumPar.put(" ", -1);
     251        final TrainingDialog editDialog = new TrainingDialog(key, row,
     252                dumPar, focusOnKey);
     253        editDialog.showDialog();
     254    }
     255
     256    /**
     257    * If during last editProperty call user changed the key name, this key will be returned
     258    * Elsewhere, returns null.
     259    * @return The modified key, or {@code null}
     260    */
     261    public String getChangedKey() {
     262        return changedKey;
     263    }
     264
     265    /**
     266    * For a given key k, return a list of keys which are used as keys for
     267    * auto-completing values to increase the search space.
     268    * @param key the key k
     269    * @return a list of keys
     270    */
     271    private static List<String> getAutocompletionKeys(String key) {
     272        if ("name".equals(key) || "addr:street".equals(key))
     273            return Arrays.asList("addr:street", "name");
     274        else
     275            return Arrays.asList(key);
     276    }
     277
     278    /**
     279    * Load recently used tags from preferences if needed.
     280    */
     281    public void loadTagsIfNeeded() {
     282        if (PROPERTY_REMEMBER_TAGS.get() && recentTags.isEmpty()) {
     283            recentTags.clear();
     284            Collection<String> c = Main.pref.getCollection("properties.recent-tags");
     285            Iterator<String> it = c.iterator();
     286            String key, value;
     287            while (it.hasNext()) {
     288                key = it.next();
     289                value = it.next();
     290                recentTags.put(new Tag(key, value), null);
     291            }
     292        }
     293    }
     294
     295    /**
     296    * Store recently used tags in preferences if needed.
     297    */
     298    public void saveTagsIfNeeded() {
     299        if (PROPERTY_REMEMBER_TAGS.get() && !recentTags.isEmpty()) {
     300            List<String> c = new ArrayList<>(recentTags.size()*2);
     301            for (Tag t: recentTags.keySet()) {
     302                c.add(t.getKey());
     303                c.add(t.getValue());
     304            }
     305            Main.pref.putCollection("properties.recent-tags", c);
     306        }
     307    }
     308
     309    /**
     310    * Warns user about a key being overwritten.
     311    * @param action The action done by the user. Must state what key is changed
     312    * @param togglePref  The preference to save the checkbox state to
     313    * @return {@code true} if the user accepts to overwrite key, {@code false} otherwise
     314    */
     315    private boolean warnOverwriteKey(String action, String togglePref) {
     316        ExtendedDialog ed = new ExtendedDialog(
     317                Main.parent,
     318                tr("Overwrite key"),
     319                new String[]{tr("Replace"), tr("Cancel")});
     320        ed.setButtonIcons(new String[]{"purge", "cancel"});
     321        ed.setContent(action+"\n"+ tr("The new key is already used, overwrite values?"));
     322        ed.setCancelButton(2);
     323        ed.toggleEnable(togglePref);
     324        ed.showDialog();
     325
     326        return ed.getValue() == 1;
     327    }
     328
     329    public final class TrainingDialog extends AbstractTagsDialog {
     330
     331        private static final int FIELD_COLUMNS = 4;
     332        private final JTextField inputFileField;
     333        private final JLabel inputFileLabel;
     334        private final JTextField topKField;
     335        private final JTextField cParameterField;
     336        private final JTextField frequencyField;
     337
     338        private final JButton fileBrowseButton;
     339        private final JButton acceptConfigButton;
     340        private JRadioButton frequencyButton;
     341        private JRadioButton topKButton;
     342        private JCheckBox cParameterCheckBox;
     343        private final JButton resetConfigButton;
     344        private String inputFileValue;
     345        private Double cParameterValue = 0.0;
     346        private Integer topKvalue = 0;
     347        private Integer frequencyValue = 0;
     348        private boolean crossValidateFlag;
     349        private final JButton startTrainingButton;
     350        private final JLabel cErrorMessageLabel;
     351        private final JLabel topKErrorMessageLabel;
     352        private final JLabel inputFileErrorMessageLabel;
     353        private final JLabel frequencyErrorMessageLabel;
     354        private final JProgressBar trainingProgressBar;
     355        private final JRadioButton byAreaRadioButton;
     356        private final JRadioButton byTimeRadioButton;
     357        private final JLabel userNameLabel;
     358        private final JTextField userNameField;
     359        private final JTextField daysField;
     360        private final JLabel daysLabel;
     361        private final JCheckBox trainFromUserCheckBox;
     362        private final JPanel userHistoryPanel;
     363        private Integer daysValue;
     364        private String usernameValue;
     365        private TrainWorker trainWorker;
     366        private UserDataExtractAndTrainWorker userDataExtractAndTrainWorker;
     367
     368        private TrainingDialog(String key, int row, Map<String, Integer> map, final boolean initialFocusOnKey) {
     369            super(Main.parent, tr("Training process configuration"), new String[] {tr("Cancel")});
     370
     371            setButtonIcons(new String[] {"ok", "cancel"});
     372            setCancelButton(2);
     373
     374            JPanel mainPanel = new JPanel(new BorderLayout(10, 10));   //6,6
     375            JPanel configPanel = new JPanel(new BorderLayout(10, 10));  //6,6    //at NORTH of mainPanel
     376            JPanel inputPanel = new JPanel(new FlowLayout(FlowLayout.LEFT));    //NORTH at config panel
     377            JPanel paramPanel = new JPanel(new FlowLayout(FlowLayout.LEFT));    //WEST at config panel //config panel has EAST free
     378
     379            JPanel southPanel = new JPanel(new FlowLayout(FlowLayout.LEFT));    //SOUTH at config panel
     380            userHistoryPanel = new JPanel(); //CENTER of config
     381
     382            trainingProgressBar = new JProgressBar(0, 100);
     383
     384            ButtonGroup paramGroup = new ButtonGroup();
     385            ButtonGroup userGroup = new ButtonGroup();
     386            inputFileLabel = new JLabel();
     387            inputFileField = new JTextField();
     388            cParameterField = new JTextField();
     389            topKField = new JTextField();
     390            frequencyField = new JTextField();
     391
     392            cParameterCheckBox = new JCheckBox("Define C parameter");
     393            topKButton = new JRadioButton("Top-K terms");
     394            frequencyButton = new JRadioButton("Max-Frequency");
     395            fileBrowseButton = new JButton();
     396            acceptConfigButton = new JButton("Accept Configuration");
     397            resetConfigButton = new JButton("Reset Configuration/Cancel training");
     398            startTrainingButton = new JButton("Train Model");
     399
     400            inputFileErrorMessageLabel = new JLabel("");
     401            cErrorMessageLabel = new JLabel("");
     402            topKErrorMessageLabel = new JLabel("");
     403            frequencyErrorMessageLabel = new JLabel("");
     404
     405            trainFromUserCheckBox = new JCheckBox("Train Model From User");
     406            byAreaRadioButton = new JRadioButton("By Area");
     407            byTimeRadioButton = new JRadioButton("By Time");
     408            userNameLabel = new JLabel("Username:");
     409            userNameField = new JTextField();
     410
     411            daysLabel = new JLabel("Days: ");
     412            daysField = new JTextField();
     413
     414            cParameterCheckBox.setSelected(true);
     415            userHistoryPanel.setEnabled(false);
     416            byAreaRadioButton.setEnabled(false);
     417            byAreaRadioButton.setSelected(true);
     418            byTimeRadioButton.setEnabled(false);
     419            userNameLabel.setEnabled(false);
     420            userNameField.setEnabled(false);
     421            daysLabel.setEnabled(false);
     422            daysField.setEnabled(false);
     423            userNameField.setColumns(FIELD_COLUMNS);
     424            daysField.setColumns(FIELD_COLUMNS);
     425
     426            Collection<String> fileHistory = Main.pref.getCollection("file-open.history");
     427            if (!fileHistory.isEmpty()) {
     428                inputFileField.setText(MAIN_PATH);
     429            }
     430
     431            fileBrowseButton.setText("...");
     432            inputFileLabel.setText("OSM filepath: ");
     433            inputFileErrorMessageLabel.setForeground(Color.RED);
     434            inputFileErrorMessageLabel.setText("");
     435            topKField.setText("50");
     436            frequencyField.setText("200");
     437            cParameterField.setText("0.01");
     438
     439            cParameterField.setColumns(FIELD_COLUMNS);
     440            cParameterField.setEditable(false);
     441            cParameterField.setHorizontalAlignment(javax.swing.SwingConstants.RIGHT);
     442            cErrorMessageLabel.setForeground(Color.RED);
     443            cErrorMessageLabel.setMinimumSize(new Dimension(150, 10));
     444
     445            topKButton.setSelected(true);
     446            topKField.setColumns(FIELD_COLUMNS);
     447            topKField.setHorizontalAlignment(javax.swing.SwingConstants.RIGHT);
     448            topKErrorMessageLabel.setForeground(Color.RED);
     449
     450            frequencyField.setEditable(false);
     451            frequencyField.setColumns(FIELD_COLUMNS);
     452            frequencyField.setHorizontalAlignment(javax.swing.SwingConstants.RIGHT);
     453            frequencyErrorMessageLabel.setForeground(Color.RED);
     454
     455            fileBrowseButton.addActionListener(new java.awt.event.ActionListener() {
     456                @Override
     457                public void actionPerformed(java.awt.event.ActionEvent evt) {
     458                    inputFileChooserButtonActionPerformed(evt);
     459                }
     460            });
     461            topKButton.addActionListener(new java.awt.event.ActionListener() {
     462                @Override
     463                public void actionPerformed(java.awt.event.ActionEvent evt) {
     464                    if (topKButton.isSelected()) {
     465                        frequencyField.setEditable(false);
     466                        topKField.setEditable(true);
     467                    } else {
     468                        frequencyField.setEditable(true);
     469                        topKField.setEditable(false);
     470                    }
     471                }
     472            });
     473            frequencyButton.addActionListener(new java.awt.event.ActionListener() {
     474                @Override
     475                public void actionPerformed(java.awt.event.ActionEvent evt) {
     476                    if (frequencyButton.isSelected()) {
     477                        topKField.setEditable(false);
     478                        frequencyField.setEditable(true);
     479                    } else {
     480                        topKField.setEditable(true);
     481                        frequencyField.setEditable(false);
     482                    }
     483                }
     484            });
     485            cParameterCheckBox.addActionListener(new java.awt.event.ActionListener() {
     486                @Override
     487                public void actionPerformed(java.awt.event.ActionEvent evt) {
     488                    if (cParameterCheckBox.isSelected()) {
     489                        cParameterField.setEditable(true);
     490                    } else {
     491                        cParameterField.setEditable(false);
     492                    }
     493                }
     494            });
     495            acceptConfigButton.addActionListener(new java.awt.event.ActionListener() {
     496                @Override
     497                public void actionPerformed(java.awt.event.ActionEvent evt) {
     498                    acceptConfigButtonActionPerformed(evt);
     499                }
     500            });
     501            resetConfigButton.addActionListener(new java.awt.event.ActionListener() {
     502                @Override
     503                public void actionPerformed(java.awt.event.ActionEvent evt) {
     504                    resetConfigButtonActionPerformed();
     505                }
     506            });
     507
     508            startTrainingButton.addActionListener(new java.awt.event.ActionListener() {
     509                @Override
     510                public void actionPerformed(java.awt.event.ActionEvent evt) {
     511                    if (!acceptConfigButton.isEnabled()) {
     512                        startTraining();
     513                    } else {
     514                        System.out.println("Set values first!");
     515                    }
     516                }
     517            });
     518
     519            trainFromUserCheckBox.addActionListener(new java.awt.event.ActionListener() {
     520                @Override
     521                public void actionPerformed(java.awt.event.ActionEvent evt) {
     522                    if (trainFromUserCheckBox.isSelected()) {
     523                        userHistoryPanel.setEnabled(true);
     524                        byAreaRadioButton.setEnabled(true);
     525                        byTimeRadioButton.setEnabled(true);
     526                        userNameLabel.setEnabled(true);
     527                        userNameField.setEnabled(true);
     528                        daysLabel.setEnabled(true);
     529                        daysField.setEnabled(true);
     530                    } else {
     531                        userHistoryPanel.setEnabled(false);
     532                        byAreaRadioButton.setEnabled(false);
     533                        byTimeRadioButton.setEnabled(false);
     534                        userNameLabel.setEnabled(false);
     535                        userNameField.setEnabled(false);
     536                        daysLabel.setEnabled(false);
     537                        daysField.setEnabled(false);
     538                    }
     539                }
     540            });
     541
     542            byAreaRadioButton.addActionListener(new java.awt.event.ActionListener() {
     543                @Override
     544                public void actionPerformed(java.awt.event.ActionEvent evt) {
     545                    if (byAreaRadioButton.isSelected()) {
     546                        daysField.setEditable(false);
     547                    } else {
     548                        daysField.setEditable(true);
     549                    }
     550                }
     551            });
     552
     553            byTimeRadioButton.addActionListener(new java.awt.event.ActionListener() {
     554                @Override
     555                public void actionPerformed(java.awt.event.ActionEvent evt) {
     556                    if (byTimeRadioButton.isSelected()) {
     557                        daysField.setEditable(true);
     558                    } else {
     559                        daysField.setEditable(false);
     560                    }
     561                }
     562            });
     563
     564            //grouplayout for input panel
     565            buildInputPanelGroupLayout(inputPanel);
     566
     567            //grouplayout for param panel
     568            buildParamPanelGroupLayout(paramPanel);
     569
     570            inputPanel.add(inputFileLabel);
     571            inputPanel.add(inputFileField);
     572            inputPanel.add(fileBrowseButton);
     573            inputPanel.add(inputFileErrorMessageLabel);
     574
     575            paramGroup.add(topKButton);
     576            paramGroup.add(frequencyButton);
     577
     578            paramPanel.add(cParameterCheckBox);
     579            paramPanel.add(cParameterField);
     580            paramPanel.add(cErrorMessageLabel);
     581            paramPanel.add(topKButton);
     582            paramPanel.add(topKField);
     583            paramPanel.add(topKErrorMessageLabel);
     584            paramPanel.add(frequencyButton);
     585            paramPanel.add(frequencyField);
     586            paramPanel.add(frequencyErrorMessageLabel);
     587
     588            southPanel.add(acceptConfigButton);
     589            southPanel.add(resetConfigButton);
     590            southPanel.add(trainFromUserCheckBox);
     591
     592            userGroup.add(byAreaRadioButton);
     593            userGroup.add(byTimeRadioButton);
     594            userHistoryPanel.add(byAreaRadioButton);
     595            userHistoryPanel.add(byTimeRadioButton);
     596            userHistoryPanel.add(daysLabel);
     597            userHistoryPanel.add(daysField);
     598            userHistoryPanel.add(userNameLabel);
     599            userHistoryPanel.add(userNameField);
     600
     601            //grouplayout for user history panel
     602            /*
    603603                userNameLabel       userField
    604604                arearadiobutton
    605605                timeradiobutton     daysLabel   daysField
    606                         */
    607                         buildUserHistoryPanelGroupLayout();
    608 
    609                         configPanel.add(inputPanel, BorderLayout.NORTH);
    610                         configPanel.add(userHistoryPanel, BorderLayout.EAST);
    611                         configPanel.add(paramPanel, BorderLayout.WEST);
    612                         configPanel.add(southPanel, BorderLayout.SOUTH);
    613 
    614                         userHistoryPanel.setBorder(BorderFactory.createTitledBorder(BorderFactory
    615                                         .createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE), "Train by user History"));
    616                         paramPanel.setBorder(BorderFactory.createTitledBorder(BorderFactory
    617                                         .createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE), "SVM Configuration"));
    618                         inputPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    619                         configPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    620 
    621                         mainPanel.add(configPanel, BorderLayout.NORTH);
    622                         mainPanel.add(startTrainingButton, BorderLayout.CENTER);
    623                         mainPanel.add(trainingProgressBar, BorderLayout.SOUTH);
    624 
    625                         AutoCompletionManager autocomplete = MainApplication.getLayerManager().getEditLayer().data.getAutoCompletionManager();
    626                         List<AutoCompletionListItem> keyList = autocomplete.getKeys();
    627                         Collections.sort(keyList, defaultACItemComparator);
    628 
    629                         setContent(mainPanel, false);
    630                 }
    631 
    632                 private void buildInputPanelGroupLayout(JPanel inputPanel) {
    633                         GroupLayout inputGroupLayout = new GroupLayout(inputPanel);
    634                         inputPanel.setLayout(inputGroupLayout);
    635                         inputGroupLayout.setAutoCreateGaps(true);
    636                         inputGroupLayout.setAutoCreateContainerGaps(true);
    637 
    638                         GroupLayout.SequentialGroup inputHorGroup = inputGroupLayout.createSequentialGroup();
    639                         inputHorGroup.addGroup(inputGroupLayout.createParallelGroup().addComponent(inputFileLabel).
    640                                         addComponent(inputFileErrorMessageLabel));
    641                         inputHorGroup.addGroup(inputGroupLayout.createParallelGroup().addComponent(inputFileField));
    642                         inputHorGroup.addGroup(inputGroupLayout.createParallelGroup().addComponent(fileBrowseButton));
    643                         inputGroupLayout.setHorizontalGroup(inputHorGroup);
    644 
    645                         GroupLayout.SequentialGroup inputVerGroup = inputGroupLayout.createSequentialGroup();
    646                         inputVerGroup.addGroup(inputGroupLayout.createParallelGroup(Alignment.LEADING).addComponent(inputFileLabel).
    647                                         addComponent(inputFileField).addComponent(fileBrowseButton));
    648                         inputVerGroup.addGroup(inputGroupLayout.createParallelGroup(Alignment.LEADING).
    649                                         addComponent(inputFileErrorMessageLabel));
    650                         inputGroupLayout.setVerticalGroup(inputVerGroup);
    651                 }
    652 
    653                 private void buildParamPanelGroupLayout(JPanel paramPanel) {
    654                         GroupLayout paramGroupLayout = new GroupLayout(paramPanel);
    655                         paramPanel.setLayout(paramGroupLayout);
    656                         paramGroupLayout.setAutoCreateGaps(true);
    657                         paramGroupLayout.setAutoCreateContainerGaps(true);
    658 
    659                         GroupLayout.SequentialGroup paramHorGroup = paramGroupLayout.createSequentialGroup();
    660                         paramHorGroup.addGroup(paramGroupLayout.createParallelGroup().addComponent(topKButton).
    661                                         addComponent(frequencyButton).addComponent(cParameterCheckBox));
    662                         paramHorGroup.addGroup(paramGroupLayout.createParallelGroup().addComponent(cParameterField).
    663                                         addComponent(topKField).addComponent(frequencyField));
    664                         paramHorGroup.addGroup(paramGroupLayout.createParallelGroup().addComponent(cErrorMessageLabel).
    665                                         addComponent(topKErrorMessageLabel).addComponent(frequencyErrorMessageLabel));
    666                         paramGroupLayout.setHorizontalGroup(paramHorGroup);
    667 
    668                         GroupLayout.SequentialGroup paramVerGroup = paramGroupLayout.createSequentialGroup();
    669                         paramVerGroup.addGroup(paramGroupLayout.createParallelGroup(Alignment.BASELINE).
    670                                         addComponent(cParameterCheckBox).addComponent(cParameterField).addComponent(cErrorMessageLabel));
    671                         paramVerGroup.addGroup(paramGroupLayout.createParallelGroup(Alignment.BASELINE).addComponent(topKButton).
    672                                         addComponent(topKField).addComponent(topKErrorMessageLabel));
    673                         paramVerGroup.addGroup(paramGroupLayout.createParallelGroup(Alignment.BASELINE).
    674                                         addComponent(frequencyButton).addComponent(frequencyField).addComponent(frequencyErrorMessageLabel));
    675                         paramGroupLayout.setVerticalGroup(paramVerGroup);
    676                 }
    677 
    678                 private void buildUserHistoryPanelGroupLayout() {
    679                         GroupLayout userHistoryGroupLayout = new GroupLayout(userHistoryPanel);
    680                         userHistoryPanel.setLayout(userHistoryGroupLayout);
    681                         userHistoryGroupLayout.setAutoCreateGaps(true);
    682                         userHistoryGroupLayout.setAutoCreateContainerGaps(true);
    683                         userHistoryGroupLayout.linkSize(SwingConstants.HORIZONTAL, userNameField, daysLabel, daysField);
    684 
    685                         GroupLayout.SequentialGroup userHistoryHorGroup = userHistoryGroupLayout.createSequentialGroup();
    686 
    687                         userHistoryHorGroup.addGroup(userHistoryGroupLayout.createParallelGroup().addComponent(userNameLabel)
    688                                         .addComponent(byAreaRadioButton).addComponent(byTimeRadioButton));
    689                         userHistoryHorGroup.addGroup(userHistoryGroupLayout.createParallelGroup().addComponent(userNameField)
    690                                         .addComponent(daysLabel));
    691                         userHistoryHorGroup.addGroup(userHistoryGroupLayout.createParallelGroup().addComponent(daysField));
    692                         userHistoryGroupLayout.setHorizontalGroup(userHistoryHorGroup);
    693 
    694                         GroupLayout.SequentialGroup userHistoryVerGroup = userHistoryGroupLayout.createSequentialGroup();
    695                         userHistoryVerGroup.addGroup(userHistoryGroupLayout.createParallelGroup(Alignment.BASELINE).
    696                                         addComponent(userNameLabel).addComponent(userNameField));
    697                         userHistoryVerGroup.addGroup(userHistoryGroupLayout.createParallelGroup(Alignment.BASELINE).
    698                                         addComponent(byAreaRadioButton));
    699                         userHistoryVerGroup.addGroup(userHistoryGroupLayout.createParallelGroup(Alignment.BASELINE).
    700                                         addComponent(byTimeRadioButton).addComponent(daysLabel).addComponent(daysField));
    701                         userHistoryGroupLayout.setVerticalGroup(userHistoryVerGroup);
    702                 }
    703 
    704                 private void inputFileChooserButtonActionPerformed(ActionEvent evt) {
    705                         try {
    706                                 final File file = new File(inputFileField.getText());
    707                                 final JFileChooser fileChooser = new JFileChooser(file);
    708 
    709                                 final int returnVal = fileChooser.showOpenDialog(this);
    710                                 if (returnVal == JFileChooser.APPROVE_OPTION) {
    711                                         inputFileField.setText(fileChooser.getSelectedFile().getAbsolutePath());
    712                                 }
    713                         } catch (RuntimeException ex) {
    714                                 Logging.warn(ex);
    715                         }
    716                 }
    717 
    718                 private void acceptConfigButtonActionPerformed(ActionEvent evt) {
    719                         //parse values
    720                         inputFileValue = inputFileField.getText();
    721 
    722                         if (!new File(inputFileValue).exists()) {
    723                                 inputFileErrorMessageLabel.setText("OSM file does not exist");
    724                                 resetConfigButtonActionPerformed();
    725                                 return;
    726                         }
    727 
    728                         if (cParameterCheckBox.isSelected()) {
    729                                 String c = cParameterField.getText();
    730                                 try {
    731                                         cParameterValue = Double.parseDouble(c.replace(",", "."));
    732                                         cErrorMessageLabel.setText("");
    733                                 } catch (NumberFormatException ex) {
    734                                         cErrorMessageLabel.setText("Must be a number!");
    735                                         System.out.println("c must be a number!" + ex); //make empty textLabel beside c param to notify errors
    736                                         resetConfigButtonActionPerformed();
    737                                         return;
    738                                 }
    739                                 crossValidateFlag = false;
    740                         } else {
    741                                 crossValidateFlag = true;
    742                         }
    743 
    744                         if (topKButton.isSelected()) {
    745                                 String k = topKField.getText();
    746                                 try {
    747                                         topKvalue = Integer.parseInt(k);
    748                                         topKErrorMessageLabel.setText("");
    749                                 } catch (NumberFormatException ex) {
    750                                         topKErrorMessageLabel.setText("Must be an Integer!");
    751                                         resetConfigButtonActionPerformed();
    752                                         return;
    753                                 }
    754                         } else {
    755                                 String f = frequencyField.getText();
    756                                 try {
    757                                         frequencyValue = Integer.parseInt(f);
    758                                         frequencyErrorMessageLabel.setText("");
    759                                 } catch (NumberFormatException ex) {
    760                                         frequencyErrorMessageLabel.setText("Must be an Integer!");
    761                                         resetConfigButtonActionPerformed();
    762                                         return;
    763                                 }
    764                         }
    765 
    766                         if (trainFromUserCheckBox.isSelected()) {
    767                                 usernameValue = userNameField.getText();
    768                                 if (byTimeRadioButton.isSelected()) {
    769                                         try {
    770                                                 daysValue = Integer.parseInt(daysField.getText());
    771                                         } catch (NumberFormatException ex) {
    772                                                 daysField.setText("Integer!");
    773                                                 Logging.warn(ex);
    774                                         }
    775                                 }
    776 
    777                                 userHistoryPanel.setEnabled(false);
    778                                 byAreaRadioButton.setEnabled(false);
    779                                 byTimeRadioButton.setEnabled(false);
    780                                 userNameLabel.setEnabled(false);
    781                                 userNameField.setEnabled(false);
    782                                 daysLabel.setEnabled(false);
    783                                 daysField.setEnabled(false);
    784                         }
    785 
    786                         System.out.println("Running configuration:" + "\nC parameter: " + cParameterValue +" \ntopK: " + topKvalue
    787                                         + "\nMax Frequency: " + frequencyValue + "\nCross Validate?: " + crossValidateFlag);
    788 
    789                         trainFromUserCheckBox.setEnabled(false);
    790                         inputFileField.setEditable(false);
    791                         cParameterField.setEditable(false);
    792                         topKField.setEditable(false);
    793                         frequencyField.setEditable(false);
    794                         cParameterCheckBox.setEnabled(false);
    795                         topKButton.setEnabled(false);
    796                         frequencyButton.setEnabled(false);
    797                         acceptConfigButton.setEnabled(false);
    798                         fileBrowseButton.setEnabled(false);
    799                 }
    800 
    801                 private void resetConfigButtonActionPerformed() {
    802                         if (trainWorker != null) {
    803                                 try {
    804                                         trainWorker.cancel(true);
    805                                 } catch (CancellationException ex) {
    806                                         startTrainingButton.setEnabled(true);
    807                                         System.out.println(ex);
    808                                 }
    809                         }
    810                         if (userDataExtractAndTrainWorker != null) {
    811                                 try {
    812                                         userDataExtractAndTrainWorker.cancel(true);
    813                                 } catch (CancellationException ex) {
    814                                         startTrainingButton.setEnabled(true);
    815                                         System.out.println(ex);
    816                                 }
    817                         }
    818                         inputFileField.setEditable(true);
    819                         cParameterField.setEditable(true);
    820                         topKField.setEditable(true);
    821                         frequencyField.setEditable(true);
    822                         cParameterCheckBox.setEnabled(true);
    823                         topKButton.setEnabled(true);
    824                         frequencyButton.setEnabled(true);
    825                         acceptConfigButton.setEnabled(true);
    826                         fileBrowseButton.setEnabled(true);
    827                         trainFromUserCheckBox.setEnabled(true);
    828 
    829                         if (trainFromUserCheckBox.isSelected()) {
    830                                 userHistoryPanel.setEnabled(true);
    831                                 byAreaRadioButton.setEnabled(true);
    832                                 byTimeRadioButton.setEnabled(true);
    833                                 userNameLabel.setEnabled(true);
    834                                 userNameField.setEnabled(true);
    835                                 daysLabel.setEnabled(true);
    836                                 daysField.setEnabled(true);
    837                         }
    838                 }
    839 
    840                 private void startTraining() {
    841                         startTrainingButton.setEnabled(false);
    842 
    843                         if (trainFromUserCheckBox.isSelected()) { //if user training. train by area or days
    844                                 EventQueue.invokeLater(new Runnable() {
    845                                         @Override
    846                                         public void run() {
    847 
    848                                                 userDataExtractAndTrainWorker = new UserDataExtractAndTrainWorker(inputFileValue, usernameValue, daysValue,
    849                                                                 byAreaRadioButton.isSelected(), crossValidateFlag, cParameterValue, topKvalue, frequencyValue,
    850                                                                 topKButton.isSelected(), languageDetector);
    851 
    852                                                 userDataExtractAndTrainWorker.addPropertyChangeListener(new PropertyChangeListener() {
    853                                                         @Override
    854                                                         public void propertyChange(PropertyChangeEvent evt) {
    855                                                                 if ("progress".equals(evt.getPropertyName())) {
    856                                                                         int progress = (Integer) evt.getNewValue();
    857                                                                         trainingProgressBar.setValue(progress);
    858                                                                         if (progress == 100) {
    859                                                                                 startTrainingButton.setEnabled(true);
    860                                                                         }
    861                                                                 }
    862                                                         }
    863                                                 });
    864 
    865                                                 try {
    866                                                         System.out.println("executing userDataExtractAndTrainWorker Thread..");
    867                                                         userDataExtractAndTrainWorker.execute();
    868                                                 } catch (Exception ex) {
    869                                                         Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    870                                                 }
    871                                         }
    872                                 });
    873                         } else {
    874                                 EventQueue.invokeLater(new Runnable() {
    875                                         @Override
    876                                         public void run() {
    877                                                 trainWorker = new TrainWorker(inputFileValue, crossValidateFlag, cParameterValue, topKvalue, frequencyValue,
    878                                                                 topKButton.isSelected(), languageDetector);
    879 
    880                                                 trainWorker.addPropertyChangeListener(new PropertyChangeListener() {
    881                                                         @Override
    882                                                         public void propertyChange(PropertyChangeEvent evt) {
    883                                                                 if ("progress".equals(evt.getPropertyName())) {
    884                                                                         int progress = (Integer) evt.getNewValue();
    885                                                                         trainingProgressBar.setValue(progress);
    886                                                                         if (progress == 100) {
    887                                                                                 startTrainingButton.setEnabled(true);
    888                                                                         }
    889                                                                 }
    890                                                         }
    891                                                 });
    892 
    893                                                 try {
    894                                                         trainWorker.execute();
    895                                                 } catch (Exception ex) {
    896                                                         Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    897                                                 }
    898                                         }
    899                                 });
    900                         }
    901                 }
    902         }
    903 
    904         public static final BooleanProperty PROPERTY_FIX_TAG_LOCALE =
    905                         new BooleanProperty("properties.fix-tag-combobox-locale", false);
    906         public static final BooleanProperty PROPERTY_REMEMBER_TAGS =
    907                         new BooleanProperty("properties.remember-recently-added-tags", true);
    908         public static final IntegerProperty PROPERTY_RECENT_TAGS_NUMBER =
    909                         new IntegerProperty("properties.recently-added-tags", DEFAULT_LRU_TAGS_NUMBER);
    910 
    911         abstract static class AbstractTagsDialog extends ExtendedDialog {
    912                 AutoCompletingComboBox keys;
    913                 AutoCompletingComboBox values;
    914 
    915                 AbstractTagsDialog(Component parent, String title, String[] buttonTexts) {
    916                         super(parent, title, buttonTexts);
    917                         addMouseListener(new PopupMenuLauncher(popupMenu));
    918                 }
    919 
    920                 @Override
    921                 public void setupDialog() {
    922                         super.setupDialog();
    923                         final Dimension size = getSize();
    924                         // Set resizable only in width
    925                         setMinimumSize(size);
    926                         setPreferredSize(size);
    927                         // setMaximumSize does not work, and never worked, but still it seems not to bother Oracle to fix this 10-year-old bug
    928                         // https://bugs.openjdk.java.net/browse/JDK-6200438
    929                         // https://bugs.openjdk.java.net/browse/JDK-6464548
    930 
    931                         setRememberWindowGeometry(getClass().getName() + ".geometry",
    932                                         WindowGeometry.centerInWindow(Main.parent, size));
    933                 }
    934 
    935                 @Override
    936                 public void setVisible(boolean visible) {
    937                         // Do not want dialog to be resizable in height, as its size may increase each time because of the recently added tags
    938                         // So need to modify the stored geometry (size part only) in order to use the automatic positioning mechanism
    939                         if (visible) {
    940                                 WindowGeometry geometry = initWindowGeometry();
    941                                 Dimension storedSize = geometry.getSize();
    942                                 Dimension size = getSize();
    943                                 if (!storedSize.equals(size)) {
    944                                         if (storedSize.width < size.width) {
    945                                                 storedSize.width = size.width;
    946                                         }
    947                                         if (storedSize.height != size.height) {
    948                                                 storedSize.height = size.height;
    949                                         }
    950                                         rememberWindowGeometry(geometry);
    951                                 }
    952                                 if (keys != null) {
    953                                         keys.setFixedLocale(PROPERTY_FIX_TAG_LOCALE.get());
    954                                 }
    955                         }
    956                         super.setVisible(visible);
    957                 }
    958 
    959                 /**
    960                 * Create a focus handling adapter and apply in to the editor component of value
    961                 * autocompletion box.
    962                 * @param autocomplete Manager handling the autocompletion
    963                 * @param comparator Class to decide what values are offered on autocompletion
    964                 * @return The created adapter
    965                 */
    966                 protected FocusAdapter addFocusAdapter(final AutoCompletionManager autocomplete, final Comparator<AutoCompletionListItem> comparator) {
    967                         // get the combo box' editor component
    968                         JTextComponent editor = (JTextComponent) values.getEditor().getEditorComponent();
    969                         // Refresh the values model when focus is gained
    970                         FocusAdapter focus = new FocusAdapter() {
    971                                 @Override
    972                                 public void focusGained(FocusEvent e) {
    973                                         String key = keys.getEditor().getItem().toString();
    974 
    975                                         List<AutoCompletionListItem> valueList = autocomplete.getValues(getAutocompletionKeys(key));
    976                                         Collections.sort(valueList, comparator);
    977 
    978                                         values.setPossibleACItems(valueList);
    979                                         values.getEditor().selectAll();
    980                                 }
    981                         };
    982                         editor.addFocusListener(focus);
    983                         return focus;
    984                 }
    985 
    986                 protected JPopupMenu popupMenu = new JPopupMenu() {
    987                         JCheckBoxMenuItem fixTagLanguageCb = new JCheckBoxMenuItem(
    988                                         new AbstractAction(tr("Use English language for tag by default")) {
    989                                                 @Override
    990                                                 public void actionPerformed(ActionEvent e) {
    991                                                         boolean sel = ((JCheckBoxMenuItem) e.getSource()).getState();
    992                                                         PROPERTY_FIX_TAG_LOCALE.put(sel);
    993                                                 }
    994                                         });
    995                         {
    996                                 add(fixTagLanguageCb);
    997                                 fixTagLanguageCb.setState(PROPERTY_FIX_TAG_LOCALE.get());
    998                         }
    999                 };
    1000         }
    1001 
    1002         class ModelSettingsDialog extends JPanel {
    1003 
    1004                 private final JLabel chooseModelLabel;
    1005                 private final JButton chooseModelButton;
    1006                 private final JTextField chooseModelTextField;
    1007 
    1008                 private final DefaultListModel<String> combinationDefaultListModel = new DefaultListModel<>();
    1009                 private final JList<String> modelCombinationList = new JList<>(combinationDefaultListModel);
    1010                 private final JPanel modelCombinationPanel;
    1011                 private final JPanel weightsPanel;
    1012                 private final JCheckBox useModelCombinationCheckbox;
    1013                 private final JButton acceptWeightsButton;
    1014                 private final JButton resetWeightsButton;
    1015                 private final JButton removeSelectedModelButton;
    1016                 private Map<JTextField, String> weightFieldsAndPaths = new HashMap<>();
    1017                 private final Map<String, Double> normalizedPathsAndWeights = new HashMap<>();
    1018                 private final JOptionPane pane;
    1019                 private final JDialog dlg;
    1020                 private final JPanel mainPanel;
    1021                 private final JPanel singleSelectionPanel;
    1022                 private final JPanel setResetWeightsPanel;
    1023                 private final JScrollPane combinationScrollPane;
    1024                 private final JScrollPane singleSelectionScrollPane;
    1025                 private final TitledBorder modelTitle;
    1026                 private final TitledBorder weightTitle;
    1027                 private final TitledBorder combineTitle;
    1028                 private final Dimension singleSelectionDimension;
    1029                 private final Dimension modelCombinationDimension;
    1030                 private final Dimension mainPanelDimension;
    1031 
    1032                 ModelSettingsDialog(Collection<OsmPrimitive> sel1, final AddTagsDialog addDialog) {
    1033 
    1034                         loadPreviousCombinedSVMModel();
    1035                         singleSelectionDimension = new Dimension(470, 70);
    1036                         modelCombinationDimension = new Dimension(450, 250);
    1037                         mainPanelDimension = new Dimension(600, 350);
    1038 
    1039                         //------- <NORTH of main> ---------//
    1040                         mainPanel = new JPanel(new BorderLayout(10, 10));
    1041                         singleSelectionPanel = new JPanel(new BorderLayout(10, 10));
    1042                         setResetWeightsPanel = new JPanel();
    1043 
    1044                         chooseModelLabel = new JLabel("Choose a Model:");
    1045                         chooseModelTextField = new JTextField();
    1046                         chooseModelButton = new JButton("...");
    1047                         chooseModelTextField.setText(MODEL_PATH);
    1048 
    1049                         singleSelectionPanel.add(chooseModelLabel, BorderLayout.NORTH);
    1050                         singleSelectionPanel.add(chooseModelTextField, BorderLayout.WEST);
    1051                         singleSelectionPanel.add(chooseModelButton, BorderLayout.EAST);
    1052 
    1053                         singleSelectionScrollPane = new JScrollPane(singleSelectionPanel);
    1054                         singleSelectionScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
    1055                         singleSelectionScrollPane.setPreferredSize(singleSelectionDimension);
    1056 
    1057                         //------- </NORTH of main> ---------//
    1058 
    1059                         //------- <WEST of main> ---------//
    1060                         modelCombinationList.setFixedCellHeight(20);
    1061                         modelCombinationList.setEnabled(false);
    1062                         modelCombinationPanel = new JPanel(new BorderLayout(10, 10));
    1063 
    1064                         weightsPanel = new JPanel();
    1065                         weightsPanel.setLayout(new BoxLayout(weightsPanel, BoxLayout.Y_AXIS));
    1066                         weightsPanel.setEnabled(false);
    1067 
    1068 
    1069                         acceptWeightsButton = new JButton("Set Weights/Normalize");
    1070                         resetWeightsButton = new JButton("Reset Weights");
    1071                         removeSelectedModelButton = new JButton("Remove Selected");
    1072                         setResetWeightsPanel.add(acceptWeightsButton);
    1073                         setResetWeightsPanel.add(resetWeightsButton);
    1074                         setResetWeightsPanel.add(removeSelectedModelButton);
    1075                         removeSelectedModelButton.setEnabled(false);
    1076                         acceptWeightsButton.setEnabled(false);
    1077                         resetWeightsButton.setEnabled(false);
    1078 
    1079                         modelCombinationPanel.add(modelCombinationList, BorderLayout.CENTER);
    1080                         modelCombinationPanel.add(weightsPanel, BorderLayout.EAST);
    1081                         modelCombinationPanel.add(setResetWeightsPanel, BorderLayout.SOUTH);
    1082 
    1083                         combinationScrollPane = new JScrollPane(modelCombinationPanel);
    1084 
    1085                         combinationScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
    1086                         combinationScrollPane.setPreferredSize(modelCombinationDimension);   //new Dimension(450, 250) // w/h
    1087 
    1088                         //------- </WEST of main> ---------//
    1089 
    1090                         //------- <SOUTH of main> ---------//
    1091                         useModelCombinationCheckbox = new JCheckBox("Combine different models?");
    1092 
    1093                         //------- </SOUTH of main> ---------//
    1094 
    1095                         //------- <Borders> ---------//
    1096                         modelTitle = BorderFactory.createTitledBorder("Models");
    1097                         weightTitle = BorderFactory.createTitledBorder("W");
    1098                         combineTitle = BorderFactory.createTitledBorder("Combine Models");
    1099                         modelCombinationList.setBorder(modelTitle);
    1100                         weightsPanel.setBorder(weightTitle);
    1101 
    1102                         for (Entry<JTextField, String> entry : weightFieldsAndPaths.entrySet()) {
    1103                                 combinationDefaultListModel.addElement(entry.getValue());
    1104 
    1105                                 JTextField weightTextField = new JTextField("0.00");
    1106                                 weightTextField.setMaximumSize(new Dimension(80, 20));
    1107                                 weightsPanel.add(entry.getKey());
    1108                         }
    1109 
    1110                         //modelCombinationPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    1111                         modelCombinationPanel.setBorder(combineTitle);
    1112                         singleSelectionPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    1113                         //------- </Borders> ---------//
    1114 
    1115                         chooseModelButton.addActionListener(new java.awt.event.ActionListener() {
    1116                                 @Override
    1117                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1118                                         modelChooserButtonActionPerformed(evt);
    1119                                 }
    1120                         });
    1121 
    1122                         useModelCombinationCheckbox.addActionListener(new java.awt.event.ActionListener() {
    1123                                 @Override
    1124                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1125                                         userCombinationCheckboxActionPerformed(evt);
    1126                                 }
    1127                         });
    1128 
    1129                         acceptWeightsButton.addActionListener(new java.awt.event.ActionListener() {
    1130                                 @Override
    1131                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1132                                         acceptWeightsButtonActionPerformed(evt);
    1133                                 }
    1134                         });
    1135 
    1136                         resetWeightsButton.addActionListener(new java.awt.event.ActionListener() {
    1137                                 @Override
    1138                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1139                                         resetWeightsButtonActionPerformed(evt);
    1140                                 }
    1141                         });
    1142 
    1143                         removeSelectedModelButton.addActionListener(new java.awt.event.ActionListener() {
    1144                                 @Override
    1145                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1146                                         removeSelectedModelButtonActionPerformed(evt);
    1147                                 }
    1148                         });
    1149                         mainPanel.add(singleSelectionScrollPane, BorderLayout.NORTH);
    1150                         mainPanel.add(combinationScrollPane, BorderLayout.CENTER);
    1151                         mainPanel.add(useModelCombinationCheckbox, BorderLayout.SOUTH);
    1152 
    1153                         mainPanel.setPreferredSize(mainPanelDimension);
    1154 
    1155                         this.add(mainPanel);
    1156 
    1157                         pane = new JOptionPane(this, JOptionPane.PLAIN_MESSAGE, JOptionPane.OK_CANCEL_OPTION) {
    1158                                 @Override
    1159                                 public void setValue(Object newValue) {
    1160                                         super.setValue(newValue);
    1161                                         if (newValue instanceof Integer && (int) newValue == 0 && useModelCombinationCheckbox.isSelected()) {
    1162                                                 System.out.println("model settings button value: " + newValue);
    1163                                                 System.out.println("\nUSE COMBINED MODEL\n");
    1164                                                 useCombinedModel = true;
    1165                                                 useCustomSVMModel = false;
    1166 
    1167                                                 addDialog.loadSVMmodel();
    1168                                                 addDialog.createOSMObject(sel);
    1169                                                 saveCombinedModel();
    1170                                                 dlg.setVisible(false);
    1171                                         } else if (newValue instanceof Integer && (int) newValue == -1 && useModelCombinationCheckbox.isSelected()) {
    1172                                                 System.out.println("model settings button value: " + newValue);
    1173                                                 useCombinedModel = false;
    1174                                                 useCustomSVMModel = false;
    1175                                                 System.out.println("Use combined model");
    1176 
    1177                                                 addDialog.loadSVMmodel();
    1178                                                 addDialog.createOSMObject(sel);
    1179                                                 dlg.setVisible(false);
    1180                                         } else if (newValue instanceof Integer && (int) newValue == 0 && !useModelCombinationCheckbox.isSelected()) {
    1181                                                 System.out.println("model settings button value: " + newValue);
    1182                                                 System.out.println("Don t use combined model, use custom model");
    1183                                                 useCombinedModel = false;
    1184                                                 useCustomSVMModel = true;
    1185                                                 addDialog.loadSVMmodel();
    1186                                                 addDialog.createOSMObject(sel);
    1187                                                 dlg.setVisible(false);
    1188                                         } else if (newValue instanceof Integer && (int) newValue == -1 && !useModelCombinationCheckbox.isSelected()) {
    1189                                                 System.out.println("model settings button value: " + newValue);
    1190                                                 System.out.println("Don t use combined model, use custom model");
    1191                                                 useCombinedModel = false;
    1192                                                 useCustomSVMModel = false;
    1193                                                 addDialog.loadSVMmodel();
    1194                                                 addDialog.createOSMObject(sel);
    1195                                                 dlg.setVisible(false);
    1196                                         } else if (newValue == null || newValue.equals("uninitializedValue")) {
    1197                                                 System.out.println("uninitializedValue, do nothing");
    1198                                         }
    1199                                 }
    1200                         };
    1201 
    1202                         dlg = pane.createDialog(Main.parent, tr("Model Settings"));
    1203                         dlg.setVisible(true);
    1204                 }
    1205 
    1206                 public void makeVisible(boolean visible) {
    1207                         dlg.setVisible(true);
    1208                 }
    1209 
    1210                 private void modelChooserButtonActionPerformed(ActionEvent evt) {
    1211 
    1212                         try {
    1213                                 final File file = new File(chooseModelTextField.getText());
    1214                                 final JFileChooser fileChooser = new JFileChooser(file);
    1215 
    1216                                 final int returnVal = fileChooser.showOpenDialog(this);
    1217                                 if (returnVal == JFileChooser.APPROVE_OPTION) {
    1218                                         chooseModelTextField.setText(fileChooser.getSelectedFile().getAbsolutePath());
    1219                                         useCustomSVMModel = true;
    1220                                         customSVMModelPath = fileChooser.getSelectedFile().getAbsolutePath();
    1221                                 }
    1222 
    1223                                 if (useModelCombinationCheckbox.isSelected()) {
    1224                                         String svmModelPath = fileChooser.getSelectedFile().getAbsolutePath();
    1225                                         String svmModelText;
    1226                                         if (System.getProperty("os.name").contains("ux")) {
    1227                                                 if (svmModelPath.contains("/")) {
    1228                                                         svmModelText = svmModelPath.substring(svmModelPath.lastIndexOf("/"));
    1229                                                 } else {
    1230                                                         svmModelText = svmModelPath;
    1231                                                 }
    1232                                         } else {
    1233                                                 if (svmModelPath.contains("\\")) {
    1234                                                         svmModelText = svmModelPath.substring(svmModelPath.lastIndexOf("\\"));
    1235                                                 } else {
    1236                                                         svmModelText = svmModelPath;
    1237                                                 }
    1238                                         }
    1239                                         combinationDefaultListModel.addElement(svmModelText);
    1240                                         JTextField weightTextField = new JTextField("0.00");
    1241                                         weightFieldsAndPaths.put(weightTextField, svmModelPath);
    1242                                         System.out.println("weights size: " + weightFieldsAndPaths.size());
    1243 
    1244                                         weightTextField.setMaximumSize(new Dimension(80, 20));
    1245                                         weightsPanel.add(weightTextField);
    1246                                         //add additional textbox
    1247                                 }
    1248                         } catch (RuntimeException ex) {
    1249                                 Logging.warn(ex);
    1250                         }
    1251                 }
    1252 
    1253                 private void userCombinationCheckboxActionPerformed(java.awt.event.ActionEvent evt) {
    1254 
    1255                         if (useModelCombinationCheckbox.isSelected()) {
    1256                                 useCombinedModel = true;
    1257                                 useCustomSVMModel = false; //reseting the selected custom SVM model only here
    1258                                 removeSelectedModelButton.setEnabled(true);
    1259                                 acceptWeightsButton.setEnabled(true);
    1260                                 resetWeightsButton.setEnabled(true);
    1261 
    1262                                 chooseModelTextField.setEnabled(false);
    1263                                 modelCombinationList.setEnabled(true);
    1264                                 weightsPanel.setEnabled(true);
    1265                                 Component[] weightPanelComponents = weightsPanel.getComponents();
    1266                                 for (Component weightPanelComponent : weightPanelComponents) {
    1267                                         weightPanelComponent.setEnabled(true);
    1268                                 }
    1269                         } else {
    1270                                 useCombinedModel = false;
    1271                                 useCustomSVMModel = true;
    1272                                 removeSelectedModelButton.setEnabled(false);
    1273                                 acceptWeightsButton.setEnabled(false);
    1274                                 resetWeightsButton.setEnabled(false);
    1275 
    1276                                 chooseModelTextField.setEnabled(true);
    1277                                 modelCombinationList.setEnabled(false);
    1278                                 weightsPanel.setEnabled(false);
    1279                                 Component[] weightPanelComponents = weightsPanel.getComponents();
    1280                                 for (Component weightPanelComponent : weightPanelComponents) {
    1281                                         weightPanelComponent.setEnabled(false);
    1282                                 }
    1283                         }
    1284                 }
    1285 
    1286                 private void acceptWeightsButtonActionPerformed(ActionEvent evt) {
    1287                         int weightsCount = 0;
    1288                         removeSelectedModelButton.setEnabled(false);
    1289                         double weightSum = 0;
    1290                         for (JTextField weightField : weightFieldsAndPaths.keySet()) {
    1291                                 if (weightField.getText().equals("")) {
    1292                                         weightField.setText("0.00");
    1293                                 }
    1294 
    1295                                 try {
    1296                                         //TODO replace "," with "." to parse doubles with commas
    1297                                         Double weightValue = Double.parseDouble(weightField.getText());
    1298 
    1299                                         weightValue = Math.abs(weightValue);
    1300                                         weightSum += weightValue;
    1301                                 } catch (NumberFormatException ex) {
    1302                                         Logging.warn(ex);
    1303                                 }
    1304                                 weightsCount++;
    1305                         }
    1306 
    1307                         if (!filesAndWeights.isEmpty()) {
    1308                                 filesAndWeights.clear();
    1309                         }
    1310 
    1311                         for (JTextField weightField : weightFieldsAndPaths.keySet()) {
    1312                                 try {
    1313                                         Double weightValue = Double.parseDouble(weightField.getText());
    1314 
    1315                                         weightValue = Math.abs(weightValue)/weightSum; //normalize
    1316 
    1317                                         if (weightSum == 0) {
    1318                                                 weightValue = 1.0/weightsCount;
    1319                                         }
    1320 
    1321                                         weightField.setText(new DecimalFormat("#.##").format(weightValue));
    1322                                         normalizedPathsAndWeights.put(weightFieldsAndPaths.get(weightField), weightValue);
    1323                                         filesAndWeights.put(new File(weightFieldsAndPaths.get(weightField)), weightValue);
    1324                                         System.out.println("normalized: " + weightFieldsAndPaths.get(weightField) + "->" + weightValue);
    1325                                         weightField.setEnabled(false);
    1326 
    1327                                 } catch (NumberFormatException ex) {
    1328                                         Logging.warn(ex);
    1329                                 }
    1330                         }
    1331 
    1332                         useCombinedModel = true;
    1333                         useCustomSVMModel = false;
    1334                 }
    1335 
    1336                 private void resetWeightsButtonActionPerformed(ActionEvent evt) {
    1337                         removeSelectedModelButton.setEnabled(true);
    1338                         for (JTextField weightField : weightFieldsAndPaths.keySet()) {
    1339                                 weightField.setEnabled(true);
    1340                         }
    1341                 }
    1342 
    1343                 private void removeSelectedModelButtonActionPerformed(ActionEvent evt) {
    1344                         int index = modelCombinationList.getSelectedIndex();
    1345                         String modelToBeRemoved = combinationDefaultListModel.get(index);
    1346                         combinationDefaultListModel.remove(index);
    1347                         System.out.println("model to be removed: " + modelToBeRemoved);
    1348 
    1349                         Iterator<Entry<JTextField, String>> it = weightFieldsAndPaths.entrySet().iterator();
    1350                         while (it.hasNext()) {
    1351                                 Entry<JTextField, String> en = it.next();
    1352                                 if (en.getValue().equals(modelToBeRemoved)) {
    1353                                         it.remove();
    1354                                 }
    1355                         }
    1356                         System.out.println("model to be removed: " + modelToBeRemoved);
    1357 
    1358                         weightsPanel.remove(index);
    1359                         weightsPanel.revalidate();
    1360                         weightsPanel.repaint();
    1361                 }
    1362 
    1363                 @SuppressWarnings("unchecked")
    1364                 private void loadPreviousCombinedSVMModel() {
    1365                         File combinedModelClassesFile = new File(combinedModelClasses);
    1366 
    1367                         if (combinedModelClassesFile.exists()) {
    1368                                 FileInputStream fileIn = null;
    1369                                 ObjectInputStream in = null;
    1370                                 try {
    1371                                         fileIn = new FileInputStream(combinedModelClassesFile);
    1372                                         in = new ObjectInputStream(fileIn);
    1373                                         weightFieldsAndPaths = (Map<JTextField, String>) in.readObject();
    1374                                 } catch (FileNotFoundException ex) {
    1375                                         Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    1376                                 } catch (IOException | ClassNotFoundException ex) {
    1377                                         Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    1378                                 } finally {
    1379                                         try {
    1380                                                 if (in != null) {
    1381                                                         in.close();
    1382                                                 }
    1383                                                 if (fileIn != null) {
    1384                                                         fileIn.close();
    1385                                                 }
    1386 
    1387                                         } catch (IOException ex) {
    1388                                                 Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    1389                                         }
    1390                                 }
    1391                         } else {
    1392                                 try {
    1393                                         combinedModelClassesFile.createNewFile();
    1394                                 } catch (IOException ex) {
    1395                                         Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    1396                                 }
    1397                         }
    1398                 }
    1399 
    1400                 private void saveCombinedModel() {
    1401                         try (FileOutputStream fileOut = new FileOutputStream(combinedModelClasses);
    1402                                         ObjectOutputStream out = new ObjectOutputStream(fileOut)) {
    1403                                 out.writeObject(weightFieldsAndPaths);
    1404                         } catch (IOException e) {
    1405                                 System.out.println("serialize error" + e);
    1406                         }
    1407                 }
    1408         }
    1409 
    1410         class AddTagsDialog extends AbstractTagsDialog {
    1411                 List<JosmAction> recentTagsActions = new ArrayList<>();
    1412 
    1413                 // Counter of added commands for possible undo
    1414                 private int commandCount;
    1415                 private final JLabel recommendedClassesLabel;
    1416                 private final JButton modelSettingsButton;
    1417                 private final JButton addAndContinueButton;
    1418                 private final DefaultListModel<String> model;
    1419                 private final JList<String> categoryList;
    1420                 private Model modelSVM;
    1421                 private int modelSVMLabelSize;
    1422                 private int[] modelSVMLabels;
    1423                 private Map<String, String> mappings;
    1424                 private Map<String, Integer> mapperWithIDs;
    1425                 private Map<Integer, String> idsWithMappings;
    1426                 private List<String> textualList = new ArrayList<>();
    1427                 private final JCheckBox useTagsCheckBox;
    1428                 private ModelSettingsDialog modelSettingsDialog;
    1429                 private static final int RECOMMENDATIONS_SIZE = 10;
    1430 
    1431                 AddTagsDialog() {
    1432                         super(Main.parent, tr("Add value?"), new String[] {tr("OK"), tr("Cancel")});
    1433                         setButtonIcons(new String[] {"ok", "cancel"});
    1434                         setCancelButton(2);
    1435                         configureContextsensitiveHelp("/Dialog/AddValue", true /* show help button */);
    1436                         final AddTagsDialog addTagsDialog = this;
    1437 
    1438                         loadOntology();
    1439                         //if the user did not train a model by running the training process
    1440                         //the list does not exist in a file and so we load the default list from the jar.
    1441 
    1442                         System.out.println("path for textual: " + TEXTUAL_LIST_PATH);
    1443                         File textualListFile = new File(TEXTUAL_LIST_PATH);
    1444                         if (textualListFile.exists()) {
    1445                                 loadTextualList(textualListFile);
    1446                         } else {
    1447                                 loadDefaultTextualList();
    1448                         }
    1449 
    1450                         //if training process has not been performed, we use two sample SVM models, extracted from the jar
    1451 
    1452                         JPanel splitPanel = new JPanel(new BorderLayout(10, 10));
    1453                         JPanel mainPanel = new JPanel(new GridBagLayout()); //original panel, will be wrapped by the splitPanel
    1454                         JPanel recommendPanel = new JPanel(new BorderLayout(10, 10));   //will contain listPanel, action panel
    1455                         JPanel listPanel = new JPanel(new BorderLayout(10, 10)); //class recommend label, recommendation list
    1456                         JPanel actionsPanel = new JPanel(new FlowLayout(FlowLayout.LEFT)); //model selection buttons or configuration
    1457 
    1458                         addAndContinueButton = new JButton("Add and continue");
    1459                         modelSettingsButton = new JButton("Model Settings");
    1460                         useTagsCheckBox = new JCheckBox("Predict using tags");
    1461                         recommendedClassesLabel = new JLabel("Recommended Classes:");
    1462 
    1463                         addAndContinueButton.addActionListener(new java.awt.event.ActionListener() {
    1464                                 @Override
    1465                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1466                                         String selectedClass = categoryList.getSelectedValue();
    1467                                         addAndContinueButtonActionPerformed(evt, selectedClass);
    1468 
    1469                                         //reconstruct vector for instance and use the model that was trained with classes here
    1470 
    1471                                         List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
    1472                                         OsmPrimitive s;
    1473 
    1474                                         //get a simple selection
    1475                                         if (!osmPrimitiveSelection.isEmpty()) {
    1476                                                 s = osmPrimitiveSelection.get(0);
    1477                                                 if (s.getInterestingTags().isEmpty()) {
    1478                                                         //load original model
    1479                                                         modelWithClasses = false;
    1480                                                         loadSVMmodel();
    1481                                                         createOSMObject(sel); //create object without class features
    1482                                                 } else {
    1483                                                         //recommend using tags: set the checkbox selected to avoid confusing the user
    1484                                                         useTagsCheckBox.setSelected(true);
    1485 
    1486                                                         if (useTagsCheckBox.isSelected()) {
    1487                                                                 //load model with classes
    1488                                                                 modelWithClasses = true;
    1489                                                                 loadSVMmodel();
    1490                                                                 createOSMObject(sel); //create object including class features
    1491                                                         } else {
    1492                                                                 modelWithClasses = false;
    1493                                                                 loadSVMmodel();
    1494                                                                 createOSMObject(sel); //create object including class features
    1495                                                         }
    1496                                                 }
    1497                                         }
    1498                                 }
    1499                         });
    1500 
    1501                         modelSettingsButton.addActionListener(new java.awt.event.ActionListener() {
    1502                                 @Override
    1503                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1504                                         if (modelSettingsDialog == null) {
    1505                                                 System.out.println("new modelSettingsDialog");
    1506                                                 modelSettingsDialog = new ModelSettingsDialog(sel, addTagsDialog);
    1507                                         } else {
    1508                                                 System.out.println("set modelSettingsDialog visible");
    1509                                                 modelSettingsDialog.makeVisible(true);
    1510                                         }
    1511                                 }
    1512                         });
    1513 
    1514                         useTagsCheckBox.addActionListener(new java.awt.event.ActionListener() {
    1515                                 @Override
    1516                                 public void actionPerformed(java.awt.event.ActionEvent evt) {
    1517                                         List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
    1518                                         OsmPrimitive s;
    1519                                         if (!osmPrimitiveSelection.isEmpty()) {
    1520                                                 s = osmPrimitiveSelection.get(0);
    1521                                                 if (s.getInterestingTags().isEmpty()) {
    1522                                                         //load original model
    1523                                                         modelWithClasses = false;
    1524                                                         loadSVMmodel();
    1525                                                         createOSMObject(sel); //create object without class features
    1526                                                 } else {
    1527                                                         //useTagsCheckBox
    1528                                                         if (useTagsCheckBox.isSelected()) {
    1529                                                                 //load model with classes
    1530                                                                 modelWithClasses = true;
    1531                                                                 loadSVMmodel();
    1532                                                                 createOSMObject(sel); //create object including class features
    1533                                                         } else {
    1534                                                                 modelWithClasses = false;
    1535                                                                 loadSVMmodel();
    1536                                                                 createOSMObject(sel); //create object including class features
    1537                                                         }
    1538                                                 }
    1539                                         }
    1540                                 }
    1541                         });
    1542 
    1543                         keys = new AutoCompletingComboBox();
    1544                         values = new AutoCompletingComboBox();
    1545 
    1546                         mainPanel.add(new JLabel("<html>"+trn("This will change up to {0} object.",
    1547                                         "This will change up to {0} objects.", sel.size(), sel.size())
    1548                         +"<br><br>"+tr("Please select a key")), GBC.eol().fill(GBC.HORIZONTAL));
    1549 
    1550                         AutoCompletionManager autocomplete = MainApplication.getLayerManager().getEditLayer().data.getAutoCompletionManager();
    1551                         List<AutoCompletionListItem> keyList = autocomplete.getKeys();
    1552 
    1553                         AutoCompletionListItem itemToSelect = null;
    1554                         // remove the object's tag keys from the list
    1555                         Iterator<AutoCompletionListItem> iter = keyList.iterator();
    1556                         while (iter.hasNext()) {
    1557                                 AutoCompletionListItem item = iter.next();
    1558                                 if (item.getValue().equals(lastAddKey)) {
    1559                                         itemToSelect = item;
    1560                                 }
    1561                                 for (int i = 0; i < tagData.getRowCount(); ++i) {
    1562                                         if (item.getValue().equals(tagData.getValueAt(i, 0))) {
    1563                                                 if (itemToSelect == item) {
    1564                                                         itemToSelect = null;
    1565                                                 }
    1566                                                 iter.remove();
    1567                                                 break;
    1568                                         }
    1569                                 }
    1570                         }
    1571 
    1572                         Collections.sort(keyList, defaultACItemComparator);
    1573                         keys.setPossibleACItems(keyList);
    1574                         keys.setEditable(true);
    1575 
    1576                         mainPanel.add(keys, GBC.eop().fill());
    1577                         mainPanel.add(new JLabel(tr("Please select a value")), GBC.eol());
    1578 
    1579                         model = new DefaultListModel<>();
    1580 
    1581                         parseTagsMappedToClasses();
    1582 
    1583                         List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
    1584                         OsmPrimitive s;
    1585                         //get a simple selection
    1586                         if (!osmPrimitiveSelection.isEmpty()) {
    1587                                 s = osmPrimitiveSelection.get(0);
    1588                                 File modelDirectory = new File(MODEL_PATH);
    1589                                 String modelWithClassesPath = modelDirectory.getAbsolutePath() + "/model_with_classes";
    1590                                 File modelWithClassesFile = new File(modelWithClassesPath);
    1591                                 if (s.getInterestingTags().isEmpty() || !modelWithClassesFile.exists()) {
    1592                                         modelWithClasses = false;
    1593                                         loadSVMmodel(); //load original model
    1594                                         createOSMObject(sel); //create object without class features
    1595                                 } else {
    1596                                         //recommend using tags: set the checkbox selected to avoid confusing the user
    1597                                         useTagsCheckBox.setSelected(true);
    1598                                         modelWithClasses = true;
    1599                                         loadSVMmodel(); //load model with classes
    1600                                         createOSMObject(sel); //create object including class features
    1601                                 }
    1602                         }
    1603 
    1604                         categoryList = new JList<>(model);
    1605 
    1606                         ListSelectionListener listSelectionListener = new ListSelectionListener() {
    1607                                 @Override
    1608                                 public void valueChanged(ListSelectionEvent listSelectionEvent) {
    1609                                         if (!listSelectionEvent.getValueIsAdjusting()) { //This prevents double events
    1610 
    1611                                                 String selectedClass = categoryList.getSelectedValue();
    1612 
    1613                                                 if (selectedClass != null) { //null check, because the model is cleared after a new recommendation
    1614                                                         //tags become unselected
    1615                                                         if ((selectedClass.indexOf(" ")+1) > 0) {
    1616                                                                 //add both key + value in tags
    1617                                                                 String keyTag = selectedClass.substring(0, selectedClass.indexOf(" "));
    1618                                                                 String valueTag = selectedClass.substring(selectedClass.indexOf(" ")+1, selectedClass.length());
    1619                                                                 keys.setSelectedItem(keyTag); //adding selected tags to textBoxes
    1620                                                                 values.setSelectedItem(valueTag);
    1621                                                         } else {
    1622                                                                 //value does not have a value, add the key tag only
    1623                                                                 String keyTag = selectedClass; //test it
    1624                                                                 keys.setSelectedItem(keyTag);
    1625                                                                 values.setSelectedItem("");
    1626                                                         }
    1627                                                 }
    1628                                         }
    1629                                 }
    1630                         };
    1631 
    1632                         categoryList.addListSelectionListener(listSelectionListener);
    1633                         categoryList.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    1634                         categoryList.setModel(model);
    1635 
    1636                         values.setEditable(true);
    1637                         mainPanel.add(values, GBC.eop().fill());
    1638                         if (itemToSelect != null) {
    1639                                 keys.setSelectedItem(itemToSelect);
    1640                                 if (lastAddValue != null) {
    1641                                         values.setSelectedItem(lastAddValue);
    1642                                 }
    1643                         }
    1644 
    1645                         FocusAdapter focus = addFocusAdapter(autocomplete, defaultACItemComparator);
    1646                         // fire focus event in advance or otherwise the popup list will be too small at first
    1647                         focus.focusGained(null);
    1648 
    1649                         int recentTagsToShow = PROPERTY_RECENT_TAGS_NUMBER.get();
    1650                         if (recentTagsToShow > MAX_LRU_TAGS_NUMBER) {
    1651                                 recentTagsToShow = MAX_LRU_TAGS_NUMBER;
    1652                         }
    1653 
    1654                         suggestRecentlyAddedTags(mainPanel, recentTagsToShow, focus);
    1655 
    1656                         mainPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    1657                         listPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    1658                         splitPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
    1659 
    1660                         listPanel.add(recommendedClassesLabel, BorderLayout.NORTH);
    1661                         listPanel.add(categoryList, BorderLayout.SOUTH);
    1662                         actionsPanel.add(addAndContinueButton);
    1663                         actionsPanel.add(modelSettingsButton);
    1664                         actionsPanel.add(useTagsCheckBox);
    1665 
    1666                         recommendPanel.add(actionsPanel, BorderLayout.WEST);
    1667                         recommendPanel.add(listPanel, BorderLayout.NORTH);
    1668 
    1669                         splitPanel.add(mainPanel, BorderLayout.WEST);
    1670                         splitPanel.add(recommendPanel, BorderLayout.EAST);
    1671 
    1672                         setContent(splitPanel, false);
    1673 
    1674                         popupMenu.add(new AbstractAction(tr("Set number of recently added tags")) {
    1675                                 @Override
    1676                                 public void actionPerformed(ActionEvent e) {
    1677                                         selectNumberOfTags();
    1678                                 }
    1679                         });
    1680                         JCheckBoxMenuItem rememberLastTags = new JCheckBoxMenuItem(
    1681                                         new AbstractAction(tr("Remember last used tags after a restart")) {
    1682                                                 @Override
    1683                                                 public void actionPerformed(ActionEvent e) {
    1684                                                         boolean sel = ((JCheckBoxMenuItem) e.getSource()).getState();
    1685                                                         PROPERTY_REMEMBER_TAGS.put(sel);
    1686                                                         if (sel) saveTagsIfNeeded();
    1687                                                 }
    1688                                         });
    1689                         rememberLastTags.setState(PROPERTY_REMEMBER_TAGS.get());
    1690                         popupMenu.add(rememberLastTags);
    1691                 }
    1692 
    1693                 private void addAndContinueButtonActionPerformed(ActionEvent evt, String selectedClass) {
    1694                         performTagAdding();
    1695                 }
    1696 
    1697                 private void selectNumberOfTags() {
    1698                         String s = JOptionPane.showInputDialog(this, tr("Please enter the number of recently added tags to display"));
    1699                         if (s != null) try {
    1700                                 int v = Integer.parseInt(s);
    1701                                 if (v >= 0 && v <= MAX_LRU_TAGS_NUMBER) {
    1702                                         PROPERTY_RECENT_TAGS_NUMBER.put(v);
    1703                                         return;
    1704                                 }
    1705                         } catch (NumberFormatException ex) {
    1706                                 Logging.warn(ex);
    1707                         }
    1708                         JOptionPane.showMessageDialog(this, tr("Please enter integer number between 0 and {0}", MAX_LRU_TAGS_NUMBER));
    1709                 }
    1710 
    1711                 private void suggestRecentlyAddedTags(JPanel mainPanel, int tagsToShow, final FocusAdapter focus) {
    1712                         if (!(tagsToShow > 0 && !recentTags.isEmpty()))
    1713                                 return;
    1714 
    1715                         mainPanel.add(new JLabel(tr("Recently added tags")), GBC.eol());
    1716 
    1717                         int count = 1;
    1718                         // We store the maximum number (9) of recent tags to allow dynamic change of number of tags shown in the preferences.
    1719                         // This implies to iterate in descending order,
    1720                         // as the oldest elements will only be removed after we reach the maximum numbern and not the number of tags to show.
    1721                         // However, as Set does not allow to iterate in descending order,
    1722                         // we need to copy its elements into a List we can access in reverse order.
    1723                         List<Tag> tags = new LinkedList<>(recentTags.keySet());
    1724                         for (int i = tags.size()-1; i >= 0 && count <= tagsToShow; i--, count++) {
    1725                                 final Tag t = tags.get(i);
    1726                                 // Create action for reusing the tag, with keyboard shortcut Ctrl+(1-5)
    1727                                 String scKey = "properties:recent:"+count;
    1728                                 String scsKey = "properties:recent:shift:"+count;
    1729                                 Shortcut sc = Shortcut.registerShortcut(scKey, tr("Choose recent tag {0}", count), KeyEvent.VK_0+count, Shortcut.CTRL);
    1730                                 final JosmAction action = new JosmAction(scKey, null, tr("Use this tag again"), sc, false) {
    1731                                         @Override
    1732                                         public void actionPerformed(ActionEvent e) {
    1733                                                 keys.setSelectedItem(t.getKey());
    1734                                                 // fix #7951, #8298 - update list of values before setting value (?)
    1735                                                 focus.focusGained(null);
    1736                                                 values.setSelectedItem(t.getValue());
    1737                                         }
    1738                                 };
    1739                                 Shortcut scs = Shortcut.registerShortcut(scsKey, tr("Apply recent tag {0}", count), KeyEvent.VK_0+count, Shortcut.CTRL_SHIFT);
    1740                                 final JosmAction actionShift = new JosmAction(scsKey, null, tr("Use this tag again"), scs, false) {
    1741                                         @Override
    1742                                         public void actionPerformed(ActionEvent e) {
    1743                                                 action.actionPerformed(null);
    1744                                                 performTagAdding();
    1745                                         }
    1746                                 };
    1747                                 recentTagsActions.add(action);
    1748                                 recentTagsActions.add(actionShift);
    1749                                 disableTagIfNeeded(t, action);
    1750                                 // Find and display icon
    1751                                 ImageIcon icon = MapPaintStyles.getNodeIcon(t, false); // Filters deprecated icon
    1752                                 if (icon == null) {
    1753                                         // If no icon found in map style look at presets
    1754                                         Map<String, String> map = new HashMap<>();
    1755                                         map.put(t.getKey(), t.getValue());
    1756                                         // If still nothing display an empty icon
    1757                                         if (icon == null) {
    1758                                                 icon = new ImageIcon(new BufferedImage(16, 16, BufferedImage.TYPE_INT_ARGB));
    1759                                         }
    1760                                 }
    1761                                 GridBagConstraints gbc = new GridBagConstraints();
    1762                                 gbc.ipadx = 5;
    1763                                 mainPanel.add(new JLabel(action.isEnabled() ? icon : GuiHelper.getDisabledIcon(icon)), gbc);
    1764                                 // Create tag label
    1765                                 final String color = action.isEnabled() ? "" : "; color:gray";
    1766                                 final JLabel tagLabel = new JLabel("<html>"
    1767                                                 + "<style>td{border:1px solid gray; font-weight:normal"+color+"}</style>"
    1768                                                 + "<table><tr><td>" + XmlWriter.encode(t.toString(), true) + "</td></tr></table></html>");
    1769                                 if (action.isEnabled()) {
    1770                                         // Register action
    1771                                         mainPanel.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(sc.getKeyStroke(), scKey);
    1772                                         mainPanel.getActionMap().put(scKey, action);
    1773                                         mainPanel.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(scs.getKeyStroke(), scsKey);
    1774                                         mainPanel.getActionMap().put(scsKey, actionShift);
    1775                                         // Make the tag label clickable and set tooltip to the action description (this displays also the keyboard shortcut)
    1776                                         tagLabel.setToolTipText((String) action.getValue(Action.SHORT_DESCRIPTION));
    1777                                         tagLabel.setCursor(Cursor.getPredefinedCursor(Cursor.HAND_CURSOR));
    1778                                         tagLabel.addMouseListener(new MouseAdapter() {
    1779                                                 @Override
    1780                                                 public void mouseClicked(MouseEvent e) {
    1781                                                         action.actionPerformed(null);
    1782                                                         // add tags and close window on double-click
    1783 
    1784                                                         if (e.getClickCount() > 1) {
    1785                                                                 buttonAction(0, null); // emulate OK click and close the dialog
    1786                                                         }
    1787                                                         // add tags on Shift-Click
    1788                                                         if (e.isShiftDown()) {
    1789                                                                 performTagAdding();
    1790                                                         }
    1791                                                 }
    1792                                         });
    1793                                 } else {
    1794                                         // Disable tag label
    1795                                         tagLabel.setEnabled(false);
    1796                                         // Explain in the tooltip why
    1797                                         tagLabel.setToolTipText(tr("The key ''{0}'' is already used", t.getKey()));
    1798                                 }
    1799                                 // Finally add label to the resulting panel
    1800                                 JPanel tagPanel = new JPanel(new FlowLayout(FlowLayout.LEFT, 0, 0));
    1801                                 tagPanel.add(tagLabel);
    1802                                 mainPanel.add(tagPanel, GBC.eol().fill(GBC.HORIZONTAL));
    1803                         }
    1804                 }
    1805 
    1806                 public void destroyActions() {
    1807                         for (JosmAction action : recentTagsActions) {
    1808                                 action.destroy();
    1809                         }
    1810                 }
    1811 
    1812                 /**
    1813                 * Read tags from comboboxes and add it to all selected objects
    1814                 */
    1815                 public final void performTagAdding() {
    1816                         String key = Tag.removeWhiteSpaces(keys.getEditor().getItem().toString());
    1817                         String value = Tag.removeWhiteSpaces(values.getEditor().getItem().toString());
    1818                         if (key.isEmpty() || value.isEmpty()) return;
    1819                         for (OsmPrimitive osm: sel) {
    1820                                 String val = osm.get(key);
    1821                                 if (val != null && !val.equals(value)) {
    1822                                         if (!warnOverwriteKey(tr("You changed the value of ''{0}'' from ''{1}'' to ''{2}''.", key, val, value),
    1823                                                         "overwriteAddKey"))
    1824                                                 return;
    1825                                         break;
    1826                                 }
    1827                         }
    1828                         lastAddKey = key;
    1829                         lastAddValue = value;
    1830                         recentTags.put(new Tag(key, value), null);
    1831                         AutoCompletionManager.rememberUserInput(key, value, false);
    1832                         commandCount++;
    1833                         MainApplication.undoRedo.add(new ChangePropertyCommand(sel, key, value));
    1834                         changedKey = key;
    1835                 }
    1836 
    1837                 public void undoAllTagsAdding() {
    1838                         MainApplication.undoRedo.undo(commandCount);
    1839                 }
    1840 
    1841                 private void disableTagIfNeeded(final Tag t, final JosmAction action) {
    1842                         // Disable action if its key is already set on the object (the key being absent from the keys list for this reason
    1843                         // performing this action leads to autocomplete to the next key (see #7671 comments)
    1844                         for (int j = 0; j < tagData.getRowCount(); ++j) {
    1845                                 if (t.getKey().equals(tagData.getValueAt(j, 0))) {
    1846                                         action.setEnabled(false);
    1847                                         break;
    1848                                 }
    1849                         }
    1850                 }
    1851 
    1852                 private void loadSVMmodel() {
    1853                         File modelDirectory = new File(MODEL_PATH);
    1854                         File modelFile;
    1855                         if (useCombinedModel) {
    1856                                 if (filesAndWeights.isEmpty()) {
    1857                                         System.out.println("No models selected! Loading defaults..");
    1858                                         if (modelWithClasses) {
    1859                                                 System.out.println("Using default/last model with classes: " + modelDirectory.getAbsolutePath() + "/model_with_classes");
    1860                                                 modelFile = new File(modelDirectory.getAbsolutePath() + "/model_with_classes");
    1861                                                 try {
    1862                                                         System.out.println("try to load model: " + modelFile.getAbsolutePath());
    1863                                                         modelSVM = Model.load(modelFile);
    1864                                                         System.out.println("model loaded!");
    1865 
    1866                                                 } catch (IOException ex) {
    1867                                                         Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
    1868                                                 }
    1869                                                 modelSVMLabelSize = modelSVM.getLabels().length;
    1870                                                 modelSVMLabels = modelSVM.getLabels();
    1871                                         } else {
    1872                                                 System.out.println("Using default/last model without classes: " + modelDirectory.getAbsolutePath() + "/best_model");
    1873                                                 modelFile = new File(modelDirectory.getAbsolutePath() + "/best_model");
    1874                                                 try {
    1875                                                         System.out.println("try to load model: " + modelFile.getAbsolutePath());
    1876                                                         modelSVM = Model.load(modelFile);
    1877                                                         System.out.println("model loaded!");
    1878 
    1879                                                 } catch (IOException ex) {
    1880                                                         Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
    1881                                                 }
    1882                                                 modelSVMLabelSize = modelSVM.getLabels().length;
    1883                                                 modelSVMLabels = modelSVM.getLabels();
    1884                                         }
    1885                                 }
    1886                                 if (modelWithClasses) { //check filenames to define if model with classes is selected
    1887                                         System.out.println("Using combined model with classes");
    1888                                         useCombinedSVMmodels(sel, true);
    1889                                 } else {
    1890                                         System.out.println("Using combined model without classes");
    1891                                         useCombinedSVMmodels(sel, false);
    1892                                 }
    1893                         } else if (useCustomSVMModel) {
    1894                                 System.out.println("custom path: " + customSVMModelPath);
    1895                                 File checkExistance = new File(customSVMModelPath);
    1896                                 if (checkExistance.exists() && checkExistance.isFile()) {
    1897                                         if (modelWithClasses) {
    1898                                                 System.out.println("Using custom model with classes: ");
    1899                                                 if (customSVMModelPath.endsWith(".0")) {
    1900                                                         String customSVMModelPathWithClasses = customSVMModelPath.substring(0, customSVMModelPath.length() - 2) + ".1";
    1901 
    1902                                                         modelFile = new File(customSVMModelPathWithClasses);
    1903                                                         System.out.println(customSVMModelPathWithClasses);
    1904                                                 } else {
    1905                                                         modelFile = new File(customSVMModelPath);
    1906                                                 }
    1907                                         } else {
    1908                                                 System.out.println("Using custom model without classes");
    1909                                                 if (customSVMModelPath.endsWith(".1")) {
    1910                                                         String customSVMModelPathWithoutClasses = customSVMModelPath.substring(0, customSVMModelPath.length() - 2) + ".0";
    1911                                                         modelFile = new File(customSVMModelPathWithoutClasses);
    1912                                                         System.out.println(customSVMModelPathWithoutClasses);
    1913                                                 } else {
    1914                                                         modelFile = new File(customSVMModelPath);
    1915                                                 }
    1916                                         }
    1917                                         try {
    1918                                                 System.out.println("try to load model: " + modelFile.getAbsolutePath());
    1919                                                 modelSVM = Model.load(modelFile);
    1920                                                 System.out.println("model loaded!");
    1921 
    1922                                         } catch (IOException ex) {
    1923                                                 Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
    1924                                         }
    1925                                         modelSVMLabelSize = modelSVM.getLabels().length;
    1926                                         modelSVMLabels = modelSVM.getLabels();
    1927 
    1928                                 } else {
    1929                                         //user chose to use a custom model, but did not provide a path to a model:
    1930                                         if (modelWithClasses) {
    1931                                                 System.out.println("Using default/last model with classes");
    1932                                                 modelFile = new File(modelDirectory.getAbsolutePath() + "/model_with_classes");
    1933                                         } else {
    1934                                                 System.out.println("Using default/last model without classes");
    1935                                                 modelFile = new File(modelDirectory.getAbsolutePath() + "/best_model");
    1936                                         }
    1937 
    1938                                         try {
    1939                                                 System.out.println("try to load model: " + modelFile.getAbsolutePath());
    1940                                                 modelSVM = Model.load(modelFile);
    1941                                                 System.out.println("model loaded!");
    1942 
    1943                                         } catch (IOException ex) {
    1944                                                 Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
    1945                                         }
    1946                                         modelSVMLabelSize = modelSVM.getLabels().length;
    1947                                         modelSVMLabels = modelSVM.getLabels();
    1948 
    1949                                 }
    1950                         } else {
    1951                                 if (modelWithClasses) {
    1952                                         System.out.println("Using default/last model with classes");
    1953                                         modelFile = new File(modelDirectory.getAbsolutePath() + "/model_with_classes");
    1954                                 } else {
    1955                                         System.out.println("Using default/last model without classes");
    1956                                         modelFile = new File(modelDirectory.getAbsolutePath() + "/best_model");
    1957                                 }
    1958 
    1959                                 try {
    1960                                         System.out.println("try to load model: " + modelFile.getAbsolutePath());
    1961                                         modelSVM = Model.load(modelFile);
    1962                                         System.out.println("model loaded!");
    1963 
    1964                                 } catch (IOException ex) {
    1965                                         Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
    1966                                 }
    1967                                 modelSVMLabelSize = modelSVM.getLabels().length;
    1968                                 modelSVMLabels = modelSVM.getLabels();
    1969                         }
    1970                 }
    1971 
    1972                 private void useCombinedSVMmodels(Collection<OsmPrimitive> sel, boolean useClassFeatures) {
    1973                         System.out.println("The system will combine " + filesAndWeights.size() + " SVM models.");
    1974 
    1975                         MathTransform transform = null;
    1976                         GeometryFactory geometryFactory = new GeometryFactory();
    1977                         CoordinateReferenceSystem sourceCRS = DefaultGeographicCRS.WGS84;
    1978                         CoordinateReferenceSystem targetCRS = DefaultGeocentricCRS.CARTESIAN;
    1979                         try {
    1980                                 transform = CRS.findMathTransform(sourceCRS, targetCRS, true);
    1981 
    1982                         } catch (FactoryException ex) {
    1983                                 Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    1984                         }
    1985 
    1986                         OSMWay selectedInstance;
    1987                         List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
    1988                         OsmPrimitive s;
    1989 
    1990                         //get a simple selection
    1991                         if (!osmPrimitiveSelection.isEmpty()) {
    1992                                 s = osmPrimitiveSelection.get(0);
    1993                         } else {
    1994                                 return;
    1995                         }
    1996 
    1997                         selectedInstance = new OSMWay();
    1998                         for (Way selectedWay : s.getDataSet().getSelectedWays()) {
    1999                                 List<Node> selectedWayNodes = selectedWay.getNodes();
    2000                                 for (Node node : selectedWayNodes) {
    2001                                         node.getCoor();
    2002                                         if (node.isLatLonKnown()) {
    2003                                                 double lat = node.getCoor().lat();
    2004                                                 double lon = node.getCoor().lon();
    2005 
    2006                                                 Coordinate sourceCoordinate = new Coordinate(lon, lat);
    2007                                                 Coordinate targetGeometry = null;
    2008                                                 try {
    2009                                                         targetGeometry = JTS.transform(sourceCoordinate, null, transform);
    2010                                                 } catch (MismatchedDimensionException | TransformException ex) {
    2011                                                         Logger.getLogger(OSMParser.class.getName()).log(Level.SEVERE, null, ex);
    2012                                                 }
    2013 
    2014                                                 Geometry geom = geometryFactory.createPoint(new Coordinate(targetGeometry));
    2015                                                 selectedInstance.addNodeGeometry(geom);
    2016                                         }
    2017                                 }
    2018                         }
    2019                         Geometry fullGeom = geometryFactory.buildGeometry(selectedInstance.getNodeGeometries());
    2020                         if ((selectedInstance.getNodeGeometries().size() > 3) &&
    2021                                         selectedInstance.getNodeGeometries().get(0).equals(selectedInstance.getNodeGeometries()
    2022                                                         .get(selectedInstance.getNodeGeometries().size()-1))) {
    2023                                 //checks if the beginning and ending node are the same and the number of nodes are more than 3.
    2024                                 //the nodes must be more than 3, because jts does not allow a construction of a linear ring with less points.
    2025                                 LinearRing linear = geometryFactory.createLinearRing(fullGeom.getCoordinates());
    2026                                 Polygon poly = new Polygon(linear, null, geometryFactory);
    2027                                 selectedInstance.setGeometry(poly);
    2028 
    2029                                 System.out.println("\n\npolygon");
    2030                         } else if (selectedInstance.getNodeGeometries().size() > 1) {
    2031                                 //it is an open geometry with more than one nodes, make it linestring
    2032                                 System.out.println("\n\nlinestring");
    2033                                 LineString lineString = geometryFactory.createLineString(fullGeom.getCoordinates());
    2034                                 selectedInstance.setGeometry(lineString);
    2035                         } else { //we assume all the rest geometries are points
    2036                                 System.out.println("\n\npoint");
    2037                                 Point point = geometryFactory.createPoint(fullGeom.getCoordinate());
    2038                                 selectedInstance.setGeometry(point);
    2039                         }
    2040 
    2041                         Map<String, String> selectedTags = s.getInterestingTags();
    2042                         selectedInstance.setAllTags(selectedTags);
    2043 
    2044                         //construct vector
    2045                         if (selectedInstance != null) {
    2046                                 int id;
    2047 
    2048                                 OSMClassification classifier = new OSMClassification();
    2049                                 classifier.calculateClasses(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
    2050 
    2051                                 if (useClassFeatures) {
    2052                                         ClassFeatures classFeatures = new ClassFeatures();
    2053                                         classFeatures.createClassFeatures(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
    2054                                         id = 1422;
    2055                                 } else {
    2056                                         id = 1;
    2057                                 }
    2058 
    2059                                 GeometryFeatures geometryFeatures = new GeometryFeatures(id);
    2060                                 geometryFeatures.createGeometryFeatures(selectedInstance);
    2061                                 id = geometryFeatures.getLastID();
    2062                                 TextualFeatures textualFeatures = new TextualFeatures(id, textualList, languageDetector);
    2063                                 textualFeatures.createTextualFeatures(selectedInstance);
    2064 
    2065                                 List<FeatureNode> featureNodeList = selectedInstance.getFeatureNodeList();
    2066 
    2067                                 FeatureNode[] featureNodeArray = new FeatureNode[featureNodeList.size()];
    2068 
    2069                                 int i = 0;
    2070                                 for (FeatureNode featureNode : featureNodeList) {
    2071                                         featureNodeArray[i] = featureNode;
    2072                                         i++;
    2073                                 }
    2074                                 FeatureNode[] testInstance2 = featureNodeArray;
    2075 
    2076                                 //compute prediction list for every model
    2077                                 int[] ranks = new int[10];
    2078 
    2079                                 for (int l = 0; l < 10; l++) {
    2080                                         ranks[l] = 10-l; //init from 10 to 1
    2081                                 }
    2082 
    2083                                 Map<String, Double> scoreMap = new HashMap<>();
    2084 
    2085                                 Map<File, Double> alignedFilesAndWeights = getAlignedModels(filesAndWeights);
    2086 
    2087                                 for (File modelFile : alignedFilesAndWeights.keySet()) {
    2088 
    2089                                         try {
    2090                                                 modelSVM = Model.load(modelFile);
    2091                                         } catch (IOException ex) {
    2092                                                 Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
    2093                                         }
    2094                                         modelSVMLabelSize = modelSVM.getLabels().length;
    2095                                         modelSVMLabels = modelSVM.getLabels();
    2096 
    2097                                         Map<Integer, Integer> mapLabelsToIDs = new HashMap<>();
    2098                                         for (int h = 0; h < modelSVMLabelSize; h++) {
    2099                                                 mapLabelsToIDs.put(modelSVMLabels[h], h);
    2100                                         }
    2101                                         double[] scores = new double[modelSVMLabelSize];
    2102                                         Linear.predictValues(modelSVM, testInstance2, scores);
    2103 
    2104                                         Map<Double, Integer> scoresValues = new HashMap<>();
    2105                                         for (int h = 0; h < scores.length; h++) {
    2106                                                 scoresValues.put(scores[h], h);
    2107                                         }
    2108 
    2109                                         Arrays.sort(scores);
    2110                                         int predicted1 = modelSVMLabels[scoresValues.get(scores[scores.length-1])];
    2111                                         int predicted2 = modelSVMLabels[scoresValues.get(scores[scores.length-2])];
    2112                                         int predicted3 = modelSVMLabels[scoresValues.get(scores[scores.length-3])];
    2113                                         int predicted4 = modelSVMLabels[scoresValues.get(scores[scores.length-4])];
    2114                                         int predicted5 = modelSVMLabels[scoresValues.get(scores[scores.length-5])];
    2115                                         int predicted6 = modelSVMLabels[scoresValues.get(scores[scores.length-6])];
    2116                                         int predicted7 = modelSVMLabels[scoresValues.get(scores[scores.length-7])];
    2117                                         int predicted8 = modelSVMLabels[scoresValues.get(scores[scores.length-8])];
    2118                                         int predicted9 = modelSVMLabels[scoresValues.get(scores[scores.length-9])];
    2119                                         int predicted10 = modelSVMLabels[scoresValues.get(scores[scores.length-10])];
    2120 
    2121                                         String[] predictedTags = new String[10];
    2122                                         for (Map.Entry<String, Integer> entry : mapperWithIDs.entrySet()) {
    2123 
    2124                                                 if (entry.getValue().equals(predicted1)) {
    2125                                                         predictedTags[0] = entry.getKey();
    2126                                                 } else if (entry.getValue().equals(predicted2)) {
    2127                                                         predictedTags[1] = entry.getKey();
    2128                                                 } else if (entry.getValue().equals(predicted3)) {
    2129                                                         predictedTags[2] = entry.getKey();
    2130                                                 } else if (entry.getValue().equals(predicted4)) {
    2131                                                         predictedTags[3] = entry.getKey();
    2132                                                 } else if (entry.getValue().equals(predicted5)) {
    2133                                                         predictedTags[4] = entry.getKey();
    2134                                                 } else if (entry.getValue().equals(predicted6)) {
    2135                                                         predictedTags[5] = entry.getKey();
    2136                                                 } else if (entry.getValue().equals(predicted7)) {
    2137                                                         predictedTags[6] = entry.getKey();
    2138                                                 } else if (entry.getValue().equals(predicted8)) {
    2139                                                         predictedTags[7] = entry.getKey();
    2140                                                 } else if (entry.getValue().equals(predicted9)) {
    2141                                                         predictedTags[8] = entry.getKey();
    2142                                                 } else if (entry.getValue().equals(predicted10)) {
    2143                                                         predictedTags[9] = entry.getKey();
    2144                                                 }
    2145                                         }
    2146                                         //clearing model, to add the new computed classes in jlist
    2147                                         model.clear();
    2148                                         for (Map.Entry<String, String> tag : mappings.entrySet()) {
    2149 
    2150                                                 for (int k = 0; k < 10; k++) {
    2151                                                         if (tag.getValue().equals(predictedTags[k])) {
    2152                                                                 predictedTags[k] = tag.getKey();
    2153                                                                 model.addElement(tag.getKey());
    2154                                                         }
    2155                                                 }
    2156                                         }
    2157                                         System.out.println("combined, predicted classes: " + Arrays.toString(predictedTags));
    2158 
    2159                                         for (int r = 0; r < ranks.length; r++) {
    2160                                                 String predictedTag = predictedTags[r];
    2161                                                 Double currentWeight = alignedFilesAndWeights.get(modelFile);
    2162                                                 double finalRank = ranks[r]*currentWeight;
    2163 
    2164                                                 if (scoreMap.containsKey(predictedTag)) {
    2165                                                         Double scoreToAdd = scoreMap.get(predictedTag);
    2166                                                         scoreMap.put(predictedTag, finalRank+scoreToAdd);
    2167                                                 } else {
    2168                                                         scoreMap.put(predictedTag, finalRank);
    2169                                                 }
    2170                                                 //add final weight - predicted tag
    2171                                         }
    2172                                 } //files iter
    2173                                 model.clear();
    2174                                 List<Double> scoresList = new ArrayList<>(scoreMap.values());
    2175                                 Collections.sort(scoresList, Collections.reverseOrder());
    2176 
    2177                                 for (Double sco : scoresList) {
    2178                                         if (model.size() > 9) {
    2179                                                 break;
    2180                                         }
    2181                                         for (Map.Entry<String, Double> scoreEntry : scoreMap.entrySet()) {
    2182                                                 if (scoreEntry.getValue().equals(sco)) {
    2183                                                         model.addElement(scoreEntry.getKey());
    2184                                                 }
    2185                                         }
    2186                                 }
    2187                         }
    2188                 }
    2189 
    2190                 private void createOSMObject(Collection<OsmPrimitive> sel) {
    2191 
    2192                         MathTransform transform = null;
    2193                         GeometryFactory geometryFactory = new GeometryFactory();
    2194                         CoordinateReferenceSystem sourceCRS = DefaultGeographicCRS.WGS84;
    2195                         CoordinateReferenceSystem targetCRS = DefaultGeocentricCRS.CARTESIAN;
    2196                         try {
    2197                                 transform = CRS.findMathTransform(sourceCRS, targetCRS, true);
    2198 
    2199                         } catch (FactoryException ex) {
    2200                                 Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
    2201                         }
    2202 
    2203                         //fire an error to the user if he has multiple selection from map
    2204 
    2205                         //we consider simple (one instance) selection, so we get the first of the sel list
    2206 
    2207                         OSMWay selectedInstance;
    2208                         List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
    2209                         OsmPrimitive s;
    2210 
    2211                         //get a simple selection
    2212                         if (!osmPrimitiveSelection.isEmpty()) {
    2213                                 s = osmPrimitiveSelection.get(0);
    2214                         } else {
    2215                                 return;
    2216                         }
    2217 
    2218                         selectedInstance = new OSMWay();
    2219                         for (Way selectedWay : s.getDataSet().getSelectedWays()) {
    2220                                 List<Node> selectedWayNodes = selectedWay.getNodes();
    2221                                 for (Node node : selectedWayNodes) {
    2222                                         node.getCoor();
    2223                                         if (node.isLatLonKnown()) {
    2224                                                 double lat = node.getCoor().lat();
    2225                                                 double lon = node.getCoor().lon();
    2226 
    2227                                                 Coordinate sourceCoordinate = new Coordinate(lon, lat);
    2228                                                 Coordinate targetGeometry = null;
    2229                                                 try {
    2230                                                         targetGeometry = JTS.transform(sourceCoordinate, null, transform);
    2231                                                 } catch (MismatchedDimensionException | TransformException ex) {
    2232                                                         Logger.getLogger(OSMParser.class.getName()).log(Level.SEVERE, null, ex);
    2233                                                 }
    2234 
    2235                                                 Geometry geom = geometryFactory.createPoint(new Coordinate(targetGeometry));
    2236                                                 selectedInstance.addNodeGeometry(geom);
    2237                                         }
    2238                                 }
    2239                         }
    2240                         Geometry fullGeom = geometryFactory.buildGeometry(selectedInstance.getNodeGeometries());
    2241 
    2242                         System.out.println("number of nodes: " + selectedInstance.getNodeGeometries().size());
    2243 
    2244                         if ((selectedInstance.getNodeGeometries().size() > 3) &&
    2245                                         selectedInstance.getNodeGeometries().get(0).equals(selectedInstance.getNodeGeometries()
    2246                                                         .get(selectedInstance.getNodeGeometries().size()-1))) {
    2247                                 //checks if the beginning and ending node are the same and the number of nodes are more than 3.
    2248                                 //the nodes must be more than 3, because jts does not allow a construction of a linear ring with less points.
    2249                                 LinearRing linear = geometryFactory.createLinearRing(fullGeom.getCoordinates());
    2250                                 Polygon poly = new Polygon(linear, null, geometryFactory);
    2251                                 selectedInstance.setGeometry(poly);
    2252 
    2253                                 System.out.println("\n\npolygon");
    2254                         } else if (selectedInstance.getNodeGeometries().size() > 1) {
    2255                                 //it is an open geometry with more than one nodes, make it linestring
    2256                                 System.out.println("\n\nlinestring");
    2257                                 LineString lineString = geometryFactory.createLineString(fullGeom.getCoordinates());
    2258                                 selectedInstance.setGeometry(lineString);
    2259                         } else { //we assume all the rest geometries are points
    2260                                 System.out.println("\n\npoint");
    2261                                 Point point = geometryFactory.createPoint(fullGeom.getCoordinate());
    2262                                 selectedInstance.setGeometry(point);
    2263                         }
    2264 
    2265                         Map<String, String> selectedTags = s.getInterestingTags();
    2266                         selectedInstance.setAllTags(selectedTags);
    2267 
    2268                         //construct vector here
    2269                         if (selectedInstance != null) {
    2270                                 int id;
    2271                                 if (mappings == null) {
    2272                                         System.out.println("null mappings ERROR");
    2273                                 }
    2274 
    2275                                 OSMClassification classifier = new OSMClassification();
    2276                                 classifier.calculateClasses(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
    2277 
    2278                                 if (modelWithClasses) {
    2279                                         ClassFeatures classFeatures = new ClassFeatures();
    2280                                         classFeatures.createClassFeatures(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
    2281                                         id = 1422;
    2282                                 } else {
    2283                                         id = 1;
    2284                                 }
    2285 
    2286                                 GeometryFeatures geometryFeatures = new GeometryFeatures(id);
    2287                                 geometryFeatures.createGeometryFeatures(selectedInstance);
    2288                                 id = geometryFeatures.getLastID();
    2289                                 TextualFeatures textualFeatures = new TextualFeatures(id, textualList, languageDetector);
    2290                                 textualFeatures.createTextualFeatures(selectedInstance);
    2291 
    2292                                 List<FeatureNode> featureNodeList = selectedInstance.getFeatureNodeList();
    2293                                 System.out.println(featureNodeList);
    2294 
    2295                                 FeatureNode[] featureNodeArray = new FeatureNode[featureNodeList.size()];
    2296 
    2297                                 int i = 0;
    2298                                 for (FeatureNode featureNode : featureNodeList) {
    2299                                         featureNodeArray[i] = featureNode;
    2300                                         i++;
    2301                                 }
    2302                                 FeatureNode[] testInstance2 = featureNodeArray;
    2303 
    2304                                 Map<Integer, Integer> mapLabelsToIDs = new HashMap<>();
    2305                                 for (int h = 0; h < modelSVMLabelSize; h++) {
    2306                                         mapLabelsToIDs.put(modelSVMLabels[h], h);
    2307                                 }
    2308 
    2309                                 double[] scores = new double[modelSVMLabelSize];
    2310                                 Linear.predictValues(modelSVM, testInstance2, scores);
    2311 
    2312                                 Map<Double, Integer> scoresValues = new HashMap<>();
    2313                                 for (int h = 0; h < scores.length; h++) {
    2314                                         scoresValues.put(scores[h], h);
    2315                                 }
    2316 
    2317                                 Arrays.sort(scores);
    2318 
    2319                                 int[] preds = new int[RECOMMENDATIONS_SIZE];
    2320                                 for (int p = 0; p < RECOMMENDATIONS_SIZE; p++) {
    2321                                         preds[p] = modelSVMLabels[scoresValues.get(scores[scores.length-(p+1)])];
    2322                                 }
    2323                                 String[] predictedTags2 = new String[RECOMMENDATIONS_SIZE];
    2324 
    2325                                 for (int p = 0; p < RECOMMENDATIONS_SIZE; p++) {
    2326                                         if (idsWithMappings.containsKey(preds[p])) {
    2327                                                 predictedTags2[p] = idsWithMappings.get(preds[p]);
    2328                                         }
    2329                                 }
    2330 
    2331                                 //clearing model, to add the new computed classes in jlist
    2332                                 model.clear();
    2333                                 for (Map.Entry<String, String> tag : mappings.entrySet()) {
    2334                                         for (int k = 0; k < 10; k++) {
    2335                                                 if (tag.getValue().equals(predictedTags2[k])) {
    2336                                                         predictedTags2[k] = tag.getKey();
    2337                                                         model.addElement(tag.getKey());
    2338                                                 }
    2339                                         }
    2340                                 }
    2341                                 System.out.println("Optimized - create OSMObject, predicted classes: " + Arrays.toString(predictedTags2));
    2342                         }
    2343                 }
    2344 
    2345                 private void parseTagsMappedToClasses() {
    2346 
    2347                         InputStream tagsToClassesMapping = TrainWorker.class.getResourceAsStream("/resources/files/Map");
    2348                         Mapper mapper = new Mapper();
    2349                         try {
    2350                                 mapper.parseFile(tagsToClassesMapping);
    2351 
    2352                         } catch (FileNotFoundException ex) {
    2353                                 Logger.getLogger(Mapper.class.getName()).log(Level.SEVERE, null, ex);
    2354                         }
    2355                         mappings = mapper.getMappings();
    2356                         mapperWithIDs = mapper.getMappingsWithIDs();
    2357                         idsWithMappings = mapper.getIDsWithMappings();
    2358                 }
    2359 
    2360                 private void loadTextualList(File textualListFile) {
    2361 
    2362                         Scanner input = null;
    2363 
    2364                         try {
    2365                                 input = new Scanner(textualListFile);
    2366                         } catch (FileNotFoundException ex) {
    2367                                 Logging.warn(ex);
    2368                         }
    2369                         while (input.hasNext()) {
    2370                                 String nextLine = input.nextLine();
    2371                                 textualList.add(nextLine);
    2372                         }
    2373                         System.out.println("Textual List parsed from file successfully." + textualList);
    2374                 }
    2375 
    2376                 private void loadDefaultTextualList() {
    2377 
    2378                         InputStream textualListStream = TrainWorker.class.getResourceAsStream("/resources/files/textualList.txt");
    2379                         TextualStatistics textualStatistics = new TextualStatistics();
    2380                         textualStatistics.parseTextualList(textualListStream);
    2381                         textualList = textualStatistics.getTextualList();
    2382                         System.out.println("Default Textual List parsed from file successfully." + textualList);
    2383                 }
    2384 
    2385                 private void loadOntology() {
    2386                         InputStream ontologyStream = TrainWorker.class.getResourceAsStream("/resources/files/owl.xml");
    2387                         Ontology ontology = new Ontology(ontologyStream);
    2388                         indirectClasses = ontology.getIndirectClasses();
    2389                         indirectClassesWithIDs = ontology.getIndirectClassesIDs();
    2390                 }
    2391 
    2392                 private Map<File, Double> getAlignedModels(Map<File, Double> filesAndWeights) {
    2393                         Map<File, Double> alignedFilesAndWeights = new HashMap<>();
    2394                         if (modelWithClasses) {
    2395                                 for (Entry<File, Double> entry : filesAndWeights.entrySet()) {
    2396                                         String absolutePath = entry.getKey().getAbsolutePath();
    2397                                         if (absolutePath.endsWith(".0")) {
    2398                                                 String newPath = absolutePath.substring(0, absolutePath.length()-2) + ".1";
    2399                                                 File alignedFile = new File(newPath);
    2400                                                 if (alignedFile.exists()) {
    2401                                                         alignedFilesAndWeights.put(alignedFile, entry.getValue());
    2402                                                 }
    2403                                         } else {
    2404                                                 alignedFilesAndWeights.put(entry.getKey(), entry.getValue());
    2405                                         }
    2406                                 }
    2407                         } else {
    2408                                 for (Entry<File, Double> entry : filesAndWeights.entrySet()) {
    2409                                         String absolutePath = entry.getKey().getAbsolutePath();
    2410                                         if (absolutePath.endsWith(".1")) {
    2411                                                 String newPath = absolutePath.substring(0, absolutePath.length()-2) + ".0";
    2412                                                 File alignedFile = new File(newPath);
    2413                                                 if (alignedFile.exists()) {
    2414                                                         alignedFilesAndWeights.put(alignedFile, entry.getValue());
    2415                                                 }
    2416                                         } else {
    2417                                                 alignedFilesAndWeights.put(entry.getKey(), entry.getValue());
    2418                                         }
    2419                                 }
    2420                         }
    2421                         return alignedFilesAndWeights;
    2422                 }
    2423         }
     606            */
     607            buildUserHistoryPanelGroupLayout();
     608
     609            configPanel.add(inputPanel, BorderLayout.NORTH);
     610            configPanel.add(userHistoryPanel, BorderLayout.EAST);
     611            configPanel.add(paramPanel, BorderLayout.WEST);
     612            configPanel.add(southPanel, BorderLayout.SOUTH);
     613
     614            userHistoryPanel.setBorder(BorderFactory.createTitledBorder(BorderFactory
     615                    .createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE), "Train by user History"));
     616            paramPanel.setBorder(BorderFactory.createTitledBorder(BorderFactory
     617                    .createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE), "SVM Configuration"));
     618            inputPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     619            configPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     620
     621            mainPanel.add(configPanel, BorderLayout.NORTH);
     622            mainPanel.add(startTrainingButton, BorderLayout.CENTER);
     623            mainPanel.add(trainingProgressBar, BorderLayout.SOUTH);
     624
     625            AutoCompletionManager autocomplete = MainApplication.getLayerManager().getEditLayer().data.getAutoCompletionManager();
     626            List<AutoCompletionListItem> keyList = autocomplete.getKeys();
     627            Collections.sort(keyList, defaultACItemComparator);
     628
     629            setContent(mainPanel, false);
     630        }
     631
     632        private void buildInputPanelGroupLayout(JPanel inputPanel) {
     633            GroupLayout inputGroupLayout = new GroupLayout(inputPanel);
     634            inputPanel.setLayout(inputGroupLayout);
     635            inputGroupLayout.setAutoCreateGaps(true);
     636            inputGroupLayout.setAutoCreateContainerGaps(true);
     637
     638            GroupLayout.SequentialGroup inputHorGroup = inputGroupLayout.createSequentialGroup();
     639            inputHorGroup.addGroup(inputGroupLayout.createParallelGroup().addComponent(inputFileLabel).
     640                    addComponent(inputFileErrorMessageLabel));
     641            inputHorGroup.addGroup(inputGroupLayout.createParallelGroup().addComponent(inputFileField));
     642            inputHorGroup.addGroup(inputGroupLayout.createParallelGroup().addComponent(fileBrowseButton));
     643            inputGroupLayout.setHorizontalGroup(inputHorGroup);
     644
     645            GroupLayout.SequentialGroup inputVerGroup = inputGroupLayout.createSequentialGroup();
     646            inputVerGroup.addGroup(inputGroupLayout.createParallelGroup(Alignment.LEADING).addComponent(inputFileLabel).
     647                    addComponent(inputFileField).addComponent(fileBrowseButton));
     648            inputVerGroup.addGroup(inputGroupLayout.createParallelGroup(Alignment.LEADING).
     649                    addComponent(inputFileErrorMessageLabel));
     650            inputGroupLayout.setVerticalGroup(inputVerGroup);
     651        }
     652
     653        private void buildParamPanelGroupLayout(JPanel paramPanel) {
     654            GroupLayout paramGroupLayout = new GroupLayout(paramPanel);
     655            paramPanel.setLayout(paramGroupLayout);
     656            paramGroupLayout.setAutoCreateGaps(true);
     657            paramGroupLayout.setAutoCreateContainerGaps(true);
     658
     659            GroupLayout.SequentialGroup paramHorGroup = paramGroupLayout.createSequentialGroup();
     660            paramHorGroup.addGroup(paramGroupLayout.createParallelGroup().addComponent(topKButton).
     661                    addComponent(frequencyButton).addComponent(cParameterCheckBox));
     662            paramHorGroup.addGroup(paramGroupLayout.createParallelGroup().addComponent(cParameterField).
     663                    addComponent(topKField).addComponent(frequencyField));
     664            paramHorGroup.addGroup(paramGroupLayout.createParallelGroup().addComponent(cErrorMessageLabel).
     665                    addComponent(topKErrorMessageLabel).addComponent(frequencyErrorMessageLabel));
     666            paramGroupLayout.setHorizontalGroup(paramHorGroup);
     667
     668            GroupLayout.SequentialGroup paramVerGroup = paramGroupLayout.createSequentialGroup();
     669            paramVerGroup.addGroup(paramGroupLayout.createParallelGroup(Alignment.BASELINE).
     670                    addComponent(cParameterCheckBox).addComponent(cParameterField).addComponent(cErrorMessageLabel));
     671            paramVerGroup.addGroup(paramGroupLayout.createParallelGroup(Alignment.BASELINE).addComponent(topKButton).
     672                    addComponent(topKField).addComponent(topKErrorMessageLabel));
     673            paramVerGroup.addGroup(paramGroupLayout.createParallelGroup(Alignment.BASELINE).
     674                    addComponent(frequencyButton).addComponent(frequencyField).addComponent(frequencyErrorMessageLabel));
     675            paramGroupLayout.setVerticalGroup(paramVerGroup);
     676        }
     677
     678        private void buildUserHistoryPanelGroupLayout() {
     679            GroupLayout userHistoryGroupLayout = new GroupLayout(userHistoryPanel);
     680            userHistoryPanel.setLayout(userHistoryGroupLayout);
     681            userHistoryGroupLayout.setAutoCreateGaps(true);
     682            userHistoryGroupLayout.setAutoCreateContainerGaps(true);
     683            userHistoryGroupLayout.linkSize(SwingConstants.HORIZONTAL, userNameField, daysLabel, daysField);
     684
     685            GroupLayout.SequentialGroup userHistoryHorGroup = userHistoryGroupLayout.createSequentialGroup();
     686
     687            userHistoryHorGroup.addGroup(userHistoryGroupLayout.createParallelGroup().addComponent(userNameLabel)
     688                    .addComponent(byAreaRadioButton).addComponent(byTimeRadioButton));
     689            userHistoryHorGroup.addGroup(userHistoryGroupLayout.createParallelGroup().addComponent(userNameField)
     690                    .addComponent(daysLabel));
     691            userHistoryHorGroup.addGroup(userHistoryGroupLayout.createParallelGroup().addComponent(daysField));
     692            userHistoryGroupLayout.setHorizontalGroup(userHistoryHorGroup);
     693
     694            GroupLayout.SequentialGroup userHistoryVerGroup = userHistoryGroupLayout.createSequentialGroup();
     695            userHistoryVerGroup.addGroup(userHistoryGroupLayout.createParallelGroup(Alignment.BASELINE).
     696                    addComponent(userNameLabel).addComponent(userNameField));
     697            userHistoryVerGroup.addGroup(userHistoryGroupLayout.createParallelGroup(Alignment.BASELINE).
     698                    addComponent(byAreaRadioButton));
     699            userHistoryVerGroup.addGroup(userHistoryGroupLayout.createParallelGroup(Alignment.BASELINE).
     700                    addComponent(byTimeRadioButton).addComponent(daysLabel).addComponent(daysField));
     701            userHistoryGroupLayout.setVerticalGroup(userHistoryVerGroup);
     702        }
     703
     704        private void inputFileChooserButtonActionPerformed(ActionEvent evt) {
     705            try {
     706                final File file = new File(inputFileField.getText());
     707                final JFileChooser fileChooser = new JFileChooser(file);
     708
     709                final int returnVal = fileChooser.showOpenDialog(this);
     710                if (returnVal == JFileChooser.APPROVE_OPTION) {
     711                    inputFileField.setText(fileChooser.getSelectedFile().getAbsolutePath());
     712                }
     713            } catch (RuntimeException ex) {
     714                Logging.warn(ex);
     715            }
     716        }
     717
     718        private void acceptConfigButtonActionPerformed(ActionEvent evt) {
     719            //parse values
     720            inputFileValue = inputFileField.getText();
     721
     722            if (!new File(inputFileValue).exists()) {
     723                inputFileErrorMessageLabel.setText("OSM file does not exist");
     724                resetConfigButtonActionPerformed();
     725                return;
     726            }
     727
     728            if (cParameterCheckBox.isSelected()) {
     729                String c = cParameterField.getText();
     730                try {
     731                    cParameterValue = Double.parseDouble(c.replace(",", "."));
     732                    cErrorMessageLabel.setText("");
     733                } catch (NumberFormatException ex) {
     734                    cErrorMessageLabel.setText("Must be a number!");
     735                    System.out.println("c must be a number!" + ex); //make empty textLabel beside c param to notify errors
     736                    resetConfigButtonActionPerformed();
     737                    return;
     738                }
     739                crossValidateFlag = false;
     740            } else {
     741                crossValidateFlag = true;
     742            }
     743
     744            if (topKButton.isSelected()) {
     745                String k = topKField.getText();
     746                try {
     747                    topKvalue = Integer.parseInt(k);
     748                    topKErrorMessageLabel.setText("");
     749                } catch (NumberFormatException ex) {
     750                    topKErrorMessageLabel.setText("Must be an Integer!");
     751                    resetConfigButtonActionPerformed();
     752                    return;
     753                }
     754            } else {
     755                String f = frequencyField.getText();
     756                try {
     757                    frequencyValue = Integer.parseInt(f);
     758                    frequencyErrorMessageLabel.setText("");
     759                } catch (NumberFormatException ex) {
     760                    frequencyErrorMessageLabel.setText("Must be an Integer!");
     761                    resetConfigButtonActionPerformed();
     762                    return;
     763                }
     764            }
     765
     766            if (trainFromUserCheckBox.isSelected()) {
     767                usernameValue = userNameField.getText();
     768                if (byTimeRadioButton.isSelected()) {
     769                    try {
     770                        daysValue = Integer.parseInt(daysField.getText());
     771                    } catch (NumberFormatException ex) {
     772                        daysField.setText("Integer!");
     773                        Logging.warn(ex);
     774                    }
     775                }
     776
     777                userHistoryPanel.setEnabled(false);
     778                byAreaRadioButton.setEnabled(false);
     779                byTimeRadioButton.setEnabled(false);
     780                userNameLabel.setEnabled(false);
     781                userNameField.setEnabled(false);
     782                daysLabel.setEnabled(false);
     783                daysField.setEnabled(false);
     784            }
     785
     786            System.out.println("Running configuration:" + "\nC parameter: " + cParameterValue +" \ntopK: " + topKvalue
     787                    + "\nMax Frequency: " + frequencyValue + "\nCross Validate?: " + crossValidateFlag);
     788
     789            trainFromUserCheckBox.setEnabled(false);
     790            inputFileField.setEditable(false);
     791            cParameterField.setEditable(false);
     792            topKField.setEditable(false);
     793            frequencyField.setEditable(false);
     794            cParameterCheckBox.setEnabled(false);
     795            topKButton.setEnabled(false);
     796            frequencyButton.setEnabled(false);
     797            acceptConfigButton.setEnabled(false);
     798            fileBrowseButton.setEnabled(false);
     799        }
     800
     801        private void resetConfigButtonActionPerformed() {
     802            if (trainWorker != null) {
     803                try {
     804                    trainWorker.cancel(true);
     805                } catch (CancellationException ex) {
     806                    startTrainingButton.setEnabled(true);
     807                    System.out.println(ex);
     808                }
     809            }
     810            if (userDataExtractAndTrainWorker != null) {
     811                try {
     812                    userDataExtractAndTrainWorker.cancel(true);
     813                } catch (CancellationException ex) {
     814                    startTrainingButton.setEnabled(true);
     815                    System.out.println(ex);
     816                }
     817            }
     818            inputFileField.setEditable(true);
     819            cParameterField.setEditable(true);
     820            topKField.setEditable(true);
     821            frequencyField.setEditable(true);
     822            cParameterCheckBox.setEnabled(true);
     823            topKButton.setEnabled(true);
     824            frequencyButton.setEnabled(true);
     825            acceptConfigButton.setEnabled(true);
     826            fileBrowseButton.setEnabled(true);
     827            trainFromUserCheckBox.setEnabled(true);
     828
     829            if (trainFromUserCheckBox.isSelected()) {
     830                userHistoryPanel.setEnabled(true);
     831                byAreaRadioButton.setEnabled(true);
     832                byTimeRadioButton.setEnabled(true);
     833                userNameLabel.setEnabled(true);
     834                userNameField.setEnabled(true);
     835                daysLabel.setEnabled(true);
     836                daysField.setEnabled(true);
     837            }
     838        }
     839
     840        private void startTraining() {
     841            startTrainingButton.setEnabled(false);
     842
     843            if (trainFromUserCheckBox.isSelected()) { //if user training. train by area or days
     844                EventQueue.invokeLater(new Runnable() {
     845                    @Override
     846                    public void run() {
     847
     848                        userDataExtractAndTrainWorker = new UserDataExtractAndTrainWorker(inputFileValue, usernameValue, daysValue,
     849                                byAreaRadioButton.isSelected(), crossValidateFlag, cParameterValue, topKvalue, frequencyValue,
     850                                topKButton.isSelected(), languageDetector);
     851
     852                        userDataExtractAndTrainWorker.addPropertyChangeListener(new PropertyChangeListener() {
     853                            @Override
     854                            public void propertyChange(PropertyChangeEvent evt) {
     855                                if ("progress".equals(evt.getPropertyName())) {
     856                                    int progress = (Integer) evt.getNewValue();
     857                                    trainingProgressBar.setValue(progress);
     858                                    if (progress == 100) {
     859                                        startTrainingButton.setEnabled(true);
     860                                    }
     861                                }
     862                            }
     863                        });
     864
     865                        try {
     866                            System.out.println("executing userDataExtractAndTrainWorker Thread..");
     867                            userDataExtractAndTrainWorker.execute();
     868                        } catch (Exception ex) {
     869                            Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     870                        }
     871                    }
     872                });
     873            } else {
     874                EventQueue.invokeLater(new Runnable() {
     875                    @Override
     876                    public void run() {
     877                        trainWorker = new TrainWorker(inputFileValue, crossValidateFlag, cParameterValue, topKvalue, frequencyValue,
     878                                topKButton.isSelected(), languageDetector);
     879
     880                        trainWorker.addPropertyChangeListener(new PropertyChangeListener() {
     881                            @Override
     882                            public void propertyChange(PropertyChangeEvent evt) {
     883                                if ("progress".equals(evt.getPropertyName())) {
     884                                    int progress = (Integer) evt.getNewValue();
     885                                    trainingProgressBar.setValue(progress);
     886                                    if (progress == 100) {
     887                                        startTrainingButton.setEnabled(true);
     888                                    }
     889                                }
     890                            }
     891                        });
     892
     893                        try {
     894                            trainWorker.execute();
     895                        } catch (Exception ex) {
     896                            Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     897                        }
     898                    }
     899                });
     900            }
     901        }
     902    }
     903
     904    public static final BooleanProperty PROPERTY_FIX_TAG_LOCALE =
     905            new BooleanProperty("properties.fix-tag-combobox-locale", false);
     906    public static final BooleanProperty PROPERTY_REMEMBER_TAGS =
     907            new BooleanProperty("properties.remember-recently-added-tags", true);
     908    public static final IntegerProperty PROPERTY_RECENT_TAGS_NUMBER =
     909            new IntegerProperty("properties.recently-added-tags", DEFAULT_LRU_TAGS_NUMBER);
     910
     911    abstract static class AbstractTagsDialog extends ExtendedDialog {
     912        AutoCompletingComboBox keys;
     913        AutoCompletingComboBox values;
     914
     915        AbstractTagsDialog(Component parent, String title, String[] buttonTexts) {
     916            super(parent, title, buttonTexts);
     917            addMouseListener(new PopupMenuLauncher(popupMenu));
     918        }
     919
     920        @Override
     921        public void setupDialog() {
     922            super.setupDialog();
     923            final Dimension size = getSize();
     924            // Set resizable only in width
     925            setMinimumSize(size);
     926            setPreferredSize(size);
     927            // setMaximumSize does not work, and never worked, but still it seems not to bother Oracle to fix this 10-year-old bug
     928            // https://bugs.openjdk.java.net/browse/JDK-6200438
     929            // https://bugs.openjdk.java.net/browse/JDK-6464548
     930
     931            setRememberWindowGeometry(getClass().getName() + ".geometry",
     932                    WindowGeometry.centerInWindow(Main.parent, size));
     933        }
     934
     935        @Override
     936        public void setVisible(boolean visible) {
     937            // Do not want dialog to be resizable in height, as its size may increase each time because of the recently added tags
     938            // So need to modify the stored geometry (size part only) in order to use the automatic positioning mechanism
     939            if (visible) {
     940                WindowGeometry geometry = initWindowGeometry();
     941                Dimension storedSize = geometry.getSize();
     942                Dimension size = getSize();
     943                if (!storedSize.equals(size)) {
     944                    if (storedSize.width < size.width) {
     945                        storedSize.width = size.width;
     946                    }
     947                    if (storedSize.height != size.height) {
     948                        storedSize.height = size.height;
     949                    }
     950                    rememberWindowGeometry(geometry);
     951                }
     952                if (keys != null) {
     953                    keys.setFixedLocale(PROPERTY_FIX_TAG_LOCALE.get());
     954                }
     955            }
     956            super.setVisible(visible);
     957        }
     958
     959        /**
     960        * Create a focus handling adapter and apply in to the editor component of value
     961        * autocompletion box.
     962        * @param autocomplete Manager handling the autocompletion
     963        * @param comparator Class to decide what values are offered on autocompletion
     964        * @return The created adapter
     965        */
     966        protected FocusAdapter addFocusAdapter(final AutoCompletionManager autocomplete, final Comparator<AutoCompletionListItem> comparator) {
     967            // get the combo box' editor component
     968            JTextComponent editor = (JTextComponent) values.getEditor().getEditorComponent();
     969            // Refresh the values model when focus is gained
     970            FocusAdapter focus = new FocusAdapter() {
     971                @Override
     972                public void focusGained(FocusEvent e) {
     973                    String key = keys.getEditor().getItem().toString();
     974
     975                    List<AutoCompletionListItem> valueList = autocomplete.getValues(getAutocompletionKeys(key));
     976                    Collections.sort(valueList, comparator);
     977
     978                    values.setPossibleACItems(valueList);
     979                    values.getEditor().selectAll();
     980                }
     981            };
     982            editor.addFocusListener(focus);
     983            return focus;
     984        }
     985
     986        protected JPopupMenu popupMenu = new JPopupMenu() {
     987            JCheckBoxMenuItem fixTagLanguageCb = new JCheckBoxMenuItem(
     988                    new AbstractAction(tr("Use English language for tag by default")) {
     989                        @Override
     990                        public void actionPerformed(ActionEvent e) {
     991                            boolean sel = ((JCheckBoxMenuItem) e.getSource()).getState();
     992                            PROPERTY_FIX_TAG_LOCALE.put(sel);
     993                        }
     994                    });
     995            {
     996                add(fixTagLanguageCb);
     997                fixTagLanguageCb.setState(PROPERTY_FIX_TAG_LOCALE.get());
     998            }
     999        };
     1000    }
     1001
     1002    class ModelSettingsDialog extends JPanel {
     1003
     1004        private final JLabel chooseModelLabel;
     1005        private final JButton chooseModelButton;
     1006        private final JTextField chooseModelTextField;
     1007
     1008        private final DefaultListModel<String> combinationDefaultListModel = new DefaultListModel<>();
     1009        private final JList<String> modelCombinationList = new JList<>(combinationDefaultListModel);
     1010        private final JPanel modelCombinationPanel;
     1011        private final JPanel weightsPanel;
     1012        private final JCheckBox useModelCombinationCheckbox;
     1013        private final JButton acceptWeightsButton;
     1014        private final JButton resetWeightsButton;
     1015        private final JButton removeSelectedModelButton;
     1016        private Map<JTextField, String> weightFieldsAndPaths = new HashMap<>();
     1017        private final Map<String, Double> normalizedPathsAndWeights = new HashMap<>();
     1018        private final JOptionPane pane;
     1019        private final JDialog dlg;
     1020        private final JPanel mainPanel;
     1021        private final JPanel singleSelectionPanel;
     1022        private final JPanel setResetWeightsPanel;
     1023        private final JScrollPane combinationScrollPane;
     1024        private final JScrollPane singleSelectionScrollPane;
     1025        private final TitledBorder modelTitle;
     1026        private final TitledBorder weightTitle;
     1027        private final TitledBorder combineTitle;
     1028        private final Dimension singleSelectionDimension;
     1029        private final Dimension modelCombinationDimension;
     1030        private final Dimension mainPanelDimension;
     1031
     1032        ModelSettingsDialog(Collection<OsmPrimitive> sel1, final AddTagsDialog addDialog) {
     1033
     1034            loadPreviousCombinedSVMModel();
     1035            singleSelectionDimension = new Dimension(470, 70);
     1036            modelCombinationDimension = new Dimension(450, 250);
     1037            mainPanelDimension = new Dimension(600, 350);
     1038
     1039            //------- <NORTH of main> ---------//
     1040            mainPanel = new JPanel(new BorderLayout(10, 10));
     1041            singleSelectionPanel = new JPanel(new BorderLayout(10, 10));
     1042            setResetWeightsPanel = new JPanel();
     1043
     1044            chooseModelLabel = new JLabel("Choose a Model:");
     1045            chooseModelTextField = new JTextField();
     1046            chooseModelButton = new JButton("...");
     1047            chooseModelTextField.setText(MODEL_PATH);
     1048
     1049            singleSelectionPanel.add(chooseModelLabel, BorderLayout.NORTH);
     1050            singleSelectionPanel.add(chooseModelTextField, BorderLayout.WEST);
     1051            singleSelectionPanel.add(chooseModelButton, BorderLayout.EAST);
     1052
     1053            singleSelectionScrollPane = new JScrollPane(singleSelectionPanel);
     1054            singleSelectionScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
     1055            singleSelectionScrollPane.setPreferredSize(singleSelectionDimension);
     1056
     1057            //------- </NORTH of main> ---------//
     1058
     1059            //------- <WEST of main> ---------//
     1060            modelCombinationList.setFixedCellHeight(20);
     1061            modelCombinationList.setEnabled(false);
     1062            modelCombinationPanel = new JPanel(new BorderLayout(10, 10));
     1063
     1064            weightsPanel = new JPanel();
     1065            weightsPanel.setLayout(new BoxLayout(weightsPanel, BoxLayout.Y_AXIS));
     1066            weightsPanel.setEnabled(false);
     1067
     1068
     1069            acceptWeightsButton = new JButton("Set Weights/Normalize");
     1070            resetWeightsButton = new JButton("Reset Weights");
     1071            removeSelectedModelButton = new JButton("Remove Selected");
     1072            setResetWeightsPanel.add(acceptWeightsButton);
     1073            setResetWeightsPanel.add(resetWeightsButton);
     1074            setResetWeightsPanel.add(removeSelectedModelButton);
     1075            removeSelectedModelButton.setEnabled(false);
     1076            acceptWeightsButton.setEnabled(false);
     1077            resetWeightsButton.setEnabled(false);
     1078
     1079            modelCombinationPanel.add(modelCombinationList, BorderLayout.CENTER);
     1080            modelCombinationPanel.add(weightsPanel, BorderLayout.EAST);
     1081            modelCombinationPanel.add(setResetWeightsPanel, BorderLayout.SOUTH);
     1082
     1083            combinationScrollPane = new JScrollPane(modelCombinationPanel);
     1084
     1085            combinationScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
     1086            combinationScrollPane.setPreferredSize(modelCombinationDimension);   //new Dimension(450, 250) // w/h
     1087
     1088            //------- </WEST of main> ---------//
     1089
     1090            //------- <SOUTH of main> ---------//
     1091            useModelCombinationCheckbox = new JCheckBox("Combine different models?");
     1092
     1093            //------- </SOUTH of main> ---------//
     1094
     1095            //------- <Borders> ---------//
     1096            modelTitle = BorderFactory.createTitledBorder("Models");
     1097            weightTitle = BorderFactory.createTitledBorder("W");
     1098            combineTitle = BorderFactory.createTitledBorder("Combine Models");
     1099            modelCombinationList.setBorder(modelTitle);
     1100            weightsPanel.setBorder(weightTitle);
     1101
     1102            for (Entry<JTextField, String> entry : weightFieldsAndPaths.entrySet()) {
     1103                combinationDefaultListModel.addElement(entry.getValue());
     1104
     1105                JTextField weightTextField = new JTextField("0.00");
     1106                weightTextField.setMaximumSize(new Dimension(80, 20));
     1107                weightsPanel.add(entry.getKey());
     1108            }
     1109
     1110            //modelCombinationPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     1111            modelCombinationPanel.setBorder(combineTitle);
     1112            singleSelectionPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     1113            //------- </Borders> ---------//
     1114
     1115            chooseModelButton.addActionListener(new java.awt.event.ActionListener() {
     1116                @Override
     1117                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1118                    modelChooserButtonActionPerformed(evt);
     1119                }
     1120            });
     1121
     1122            useModelCombinationCheckbox.addActionListener(new java.awt.event.ActionListener() {
     1123                @Override
     1124                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1125                    userCombinationCheckboxActionPerformed(evt);
     1126                }
     1127            });
     1128
     1129            acceptWeightsButton.addActionListener(new java.awt.event.ActionListener() {
     1130                @Override
     1131                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1132                    acceptWeightsButtonActionPerformed(evt);
     1133                }
     1134            });
     1135
     1136            resetWeightsButton.addActionListener(new java.awt.event.ActionListener() {
     1137                @Override
     1138                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1139                    resetWeightsButtonActionPerformed(evt);
     1140                }
     1141            });
     1142
     1143            removeSelectedModelButton.addActionListener(new java.awt.event.ActionListener() {
     1144                @Override
     1145                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1146                    removeSelectedModelButtonActionPerformed(evt);
     1147                }
     1148            });
     1149            mainPanel.add(singleSelectionScrollPane, BorderLayout.NORTH);
     1150            mainPanel.add(combinationScrollPane, BorderLayout.CENTER);
     1151            mainPanel.add(useModelCombinationCheckbox, BorderLayout.SOUTH);
     1152
     1153            mainPanel.setPreferredSize(mainPanelDimension);
     1154
     1155            this.add(mainPanel);
     1156
     1157            pane = new JOptionPane(this, JOptionPane.PLAIN_MESSAGE, JOptionPane.OK_CANCEL_OPTION) {
     1158                @Override
     1159                public void setValue(Object newValue) {
     1160                    super.setValue(newValue);
     1161                    if (newValue instanceof Integer && (int) newValue == 0 && useModelCombinationCheckbox.isSelected()) {
     1162                        System.out.println("model settings button value: " + newValue);
     1163                        System.out.println("\nUSE COMBINED MODEL\n");
     1164                        useCombinedModel = true;
     1165                        useCustomSVMModel = false;
     1166
     1167                        addDialog.loadSVMmodel();
     1168                        addDialog.createOSMObject(sel);
     1169                        saveCombinedModel();
     1170                        dlg.setVisible(false);
     1171                    } else if (newValue instanceof Integer && (int) newValue == -1 && useModelCombinationCheckbox.isSelected()) {
     1172                        System.out.println("model settings button value: " + newValue);
     1173                        useCombinedModel = false;
     1174                        useCustomSVMModel = false;
     1175                        System.out.println("Use combined model");
     1176
     1177                        addDialog.loadSVMmodel();
     1178                        addDialog.createOSMObject(sel);
     1179                        dlg.setVisible(false);
     1180                    } else if (newValue instanceof Integer && (int) newValue == 0 && !useModelCombinationCheckbox.isSelected()) {
     1181                        System.out.println("model settings button value: " + newValue);
     1182                        System.out.println("Don t use combined model, use custom model");
     1183                        useCombinedModel = false;
     1184                        useCustomSVMModel = true;
     1185                        addDialog.loadSVMmodel();
     1186                        addDialog.createOSMObject(sel);
     1187                        dlg.setVisible(false);
     1188                    } else if (newValue instanceof Integer && (int) newValue == -1 && !useModelCombinationCheckbox.isSelected()) {
     1189                        System.out.println("model settings button value: " + newValue);
     1190                        System.out.println("Don t use combined model, use custom model");
     1191                        useCombinedModel = false;
     1192                        useCustomSVMModel = false;
     1193                        addDialog.loadSVMmodel();
     1194                        addDialog.createOSMObject(sel);
     1195                        dlg.setVisible(false);
     1196                    } else if (newValue == null || newValue.equals("uninitializedValue")) {
     1197                        System.out.println("uninitializedValue, do nothing");
     1198                    }
     1199                }
     1200            };
     1201
     1202            dlg = pane.createDialog(Main.parent, tr("Model Settings"));
     1203            dlg.setVisible(true);
     1204        }
     1205
     1206        public void makeVisible(boolean visible) {
     1207            dlg.setVisible(true);
     1208        }
     1209
     1210        private void modelChooserButtonActionPerformed(ActionEvent evt) {
     1211
     1212            try {
     1213                final File file = new File(chooseModelTextField.getText());
     1214                final JFileChooser fileChooser = new JFileChooser(file);
     1215
     1216                final int returnVal = fileChooser.showOpenDialog(this);
     1217                if (returnVal == JFileChooser.APPROVE_OPTION) {
     1218                    chooseModelTextField.setText(fileChooser.getSelectedFile().getAbsolutePath());
     1219                    useCustomSVMModel = true;
     1220                    customSVMModelPath = fileChooser.getSelectedFile().getAbsolutePath();
     1221                }
     1222
     1223                if (useModelCombinationCheckbox.isSelected()) {
     1224                    String svmModelPath = fileChooser.getSelectedFile().getAbsolutePath();
     1225                    String svmModelText;
     1226                    if (System.getProperty("os.name").contains("ux")) {
     1227                        if (svmModelPath.contains("/")) {
     1228                            svmModelText = svmModelPath.substring(svmModelPath.lastIndexOf("/"));
     1229                        } else {
     1230                            svmModelText = svmModelPath;
     1231                        }
     1232                    } else {
     1233                        if (svmModelPath.contains("\\")) {
     1234                            svmModelText = svmModelPath.substring(svmModelPath.lastIndexOf("\\"));
     1235                        } else {
     1236                            svmModelText = svmModelPath;
     1237                        }
     1238                    }
     1239                    combinationDefaultListModel.addElement(svmModelText);
     1240                    JTextField weightTextField = new JTextField("0.00");
     1241                    weightFieldsAndPaths.put(weightTextField, svmModelPath);
     1242                    System.out.println("weights size: " + weightFieldsAndPaths.size());
     1243
     1244                    weightTextField.setMaximumSize(new Dimension(80, 20));
     1245                    weightsPanel.add(weightTextField);
     1246                    //add additional textbox
     1247                }
     1248            } catch (RuntimeException ex) {
     1249                Logging.warn(ex);
     1250            }
     1251        }
     1252
     1253        private void userCombinationCheckboxActionPerformed(java.awt.event.ActionEvent evt) {
     1254
     1255            if (useModelCombinationCheckbox.isSelected()) {
     1256                useCombinedModel = true;
     1257                useCustomSVMModel = false; //reseting the selected custom SVM model only here
     1258                removeSelectedModelButton.setEnabled(true);
     1259                acceptWeightsButton.setEnabled(true);
     1260                resetWeightsButton.setEnabled(true);
     1261
     1262                chooseModelTextField.setEnabled(false);
     1263                modelCombinationList.setEnabled(true);
     1264                weightsPanel.setEnabled(true);
     1265                Component[] weightPanelComponents = weightsPanel.getComponents();
     1266                for (Component weightPanelComponent : weightPanelComponents) {
     1267                    weightPanelComponent.setEnabled(true);
     1268                }
     1269            } else {
     1270                useCombinedModel = false;
     1271                useCustomSVMModel = true;
     1272                removeSelectedModelButton.setEnabled(false);
     1273                acceptWeightsButton.setEnabled(false);
     1274                resetWeightsButton.setEnabled(false);
     1275
     1276                chooseModelTextField.setEnabled(true);
     1277                modelCombinationList.setEnabled(false);
     1278                weightsPanel.setEnabled(false);
     1279                Component[] weightPanelComponents = weightsPanel.getComponents();
     1280                for (Component weightPanelComponent : weightPanelComponents) {
     1281                    weightPanelComponent.setEnabled(false);
     1282                }
     1283            }
     1284        }
     1285
     1286        private void acceptWeightsButtonActionPerformed(ActionEvent evt) {
     1287            int weightsCount = 0;
     1288            removeSelectedModelButton.setEnabled(false);
     1289            double weightSum = 0;
     1290            for (JTextField weightField : weightFieldsAndPaths.keySet()) {
     1291                if (weightField.getText().equals("")) {
     1292                    weightField.setText("0.00");
     1293                }
     1294
     1295                try {
     1296                    //TODO replace "," with "." to parse doubles with commas
     1297                    Double weightValue = Double.parseDouble(weightField.getText());
     1298
     1299                    weightValue = Math.abs(weightValue);
     1300                    weightSum += weightValue;
     1301                } catch (NumberFormatException ex) {
     1302                    Logging.warn(ex);
     1303                }
     1304                weightsCount++;
     1305            }
     1306
     1307            if (!filesAndWeights.isEmpty()) {
     1308                filesAndWeights.clear();
     1309            }
     1310
     1311            for (JTextField weightField : weightFieldsAndPaths.keySet()) {
     1312                try {
     1313                    Double weightValue = Double.parseDouble(weightField.getText());
     1314
     1315                    weightValue = Math.abs(weightValue)/weightSum; //normalize
     1316
     1317                    if (weightSum == 0) {
     1318                        weightValue = 1.0/weightsCount;
     1319                    }
     1320
     1321                    weightField.setText(new DecimalFormat("#.##").format(weightValue));
     1322                    normalizedPathsAndWeights.put(weightFieldsAndPaths.get(weightField), weightValue);
     1323                    filesAndWeights.put(new File(weightFieldsAndPaths.get(weightField)), weightValue);
     1324                    System.out.println("normalized: " + weightFieldsAndPaths.get(weightField) + "->" + weightValue);
     1325                    weightField.setEnabled(false);
     1326
     1327                } catch (NumberFormatException ex) {
     1328                    Logging.warn(ex);
     1329                }
     1330            }
     1331
     1332            useCombinedModel = true;
     1333            useCustomSVMModel = false;
     1334        }
     1335
     1336        private void resetWeightsButtonActionPerformed(ActionEvent evt) {
     1337            removeSelectedModelButton.setEnabled(true);
     1338            for (JTextField weightField : weightFieldsAndPaths.keySet()) {
     1339                weightField.setEnabled(true);
     1340            }
     1341        }
     1342
     1343        private void removeSelectedModelButtonActionPerformed(ActionEvent evt) {
     1344            int index = modelCombinationList.getSelectedIndex();
     1345            String modelToBeRemoved = combinationDefaultListModel.get(index);
     1346            combinationDefaultListModel.remove(index);
     1347            System.out.println("model to be removed: " + modelToBeRemoved);
     1348
     1349            Iterator<Entry<JTextField, String>> it = weightFieldsAndPaths.entrySet().iterator();
     1350            while (it.hasNext()) {
     1351                Entry<JTextField, String> en = it.next();
     1352                if (en.getValue().equals(modelToBeRemoved)) {
     1353                    it.remove();
     1354                }
     1355            }
     1356            System.out.println("model to be removed: " + modelToBeRemoved);
     1357
     1358            weightsPanel.remove(index);
     1359            weightsPanel.revalidate();
     1360            weightsPanel.repaint();
     1361        }
     1362
     1363        @SuppressWarnings("unchecked")
     1364        private void loadPreviousCombinedSVMModel() {
     1365            File combinedModelClassesFile = new File(combinedModelClasses);
     1366
     1367            if (combinedModelClassesFile.exists()) {
     1368                FileInputStream fileIn = null;
     1369                ObjectInputStream in = null;
     1370                try {
     1371                    fileIn = new FileInputStream(combinedModelClassesFile);
     1372                    in = new ObjectInputStream(fileIn);
     1373                    weightFieldsAndPaths = (Map<JTextField, String>) in.readObject();
     1374                } catch (FileNotFoundException ex) {
     1375                    Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     1376                } catch (IOException | ClassNotFoundException ex) {
     1377                    Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     1378                } finally {
     1379                    try {
     1380                        if (in != null) {
     1381                            in.close();
     1382                        }
     1383                        if (fileIn != null) {
     1384                            fileIn.close();
     1385                        }
     1386
     1387                    } catch (IOException ex) {
     1388                        Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     1389                    }
     1390                }
     1391            } else {
     1392                try {
     1393                    combinedModelClassesFile.createNewFile();
     1394                } catch (IOException ex) {
     1395                    Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     1396                }
     1397            }
     1398        }
     1399
     1400        private void saveCombinedModel() {
     1401            try (FileOutputStream fileOut = new FileOutputStream(combinedModelClasses);
     1402                    ObjectOutputStream out = new ObjectOutputStream(fileOut)) {
     1403                out.writeObject(weightFieldsAndPaths);
     1404            } catch (IOException e) {
     1405                System.out.println("serialize error" + e);
     1406            }
     1407        }
     1408    }
     1409
     1410    class AddTagsDialog extends AbstractTagsDialog {
     1411        List<JosmAction> recentTagsActions = new ArrayList<>();
     1412
     1413        // Counter of added commands for possible undo
     1414        private int commandCount;
     1415        private final JLabel recommendedClassesLabel;
     1416        private final JButton modelSettingsButton;
     1417        private final JButton addAndContinueButton;
     1418        private final DefaultListModel<String> model;
     1419        private final JList<String> categoryList;
     1420        private Model modelSVM;
     1421        private int modelSVMLabelSize;
     1422        private int[] modelSVMLabels;
     1423        private Map<String, String> mappings;
     1424        private Map<String, Integer> mapperWithIDs;
     1425        private Map<Integer, String> idsWithMappings;
     1426        private List<String> textualList = new ArrayList<>();
     1427        private final JCheckBox useTagsCheckBox;
     1428        private ModelSettingsDialog modelSettingsDialog;
     1429        private static final int RECOMMENDATIONS_SIZE = 10;
     1430
     1431        AddTagsDialog() {
     1432            super(Main.parent, tr("Add value?"), new String[] {tr("OK"), tr("Cancel")});
     1433            setButtonIcons(new String[] {"ok", "cancel"});
     1434            setCancelButton(2);
     1435            configureContextsensitiveHelp("/Dialog/AddValue", true /* show help button */);
     1436            final AddTagsDialog addTagsDialog = this;
     1437
     1438            loadOntology();
     1439            //if the user did not train a model by running the training process
     1440            //the list does not exist in a file and so we load the default list from the jar.
     1441
     1442            System.out.println("path for textual: " + TEXTUAL_LIST_PATH);
     1443            File textualListFile = new File(TEXTUAL_LIST_PATH);
     1444            if (textualListFile.exists()) {
     1445                loadTextualList(textualListFile);
     1446            } else {
     1447                loadDefaultTextualList();
     1448            }
     1449
     1450            //if training process has not been performed, we use two sample SVM models, extracted from the jar
     1451
     1452            JPanel splitPanel = new JPanel(new BorderLayout(10, 10));
     1453            JPanel mainPanel = new JPanel(new GridBagLayout()); //original panel, will be wrapped by the splitPanel
     1454            JPanel recommendPanel = new JPanel(new BorderLayout(10, 10));   //will contain listPanel, action panel
     1455            JPanel listPanel = new JPanel(new BorderLayout(10, 10)); //class recommend label, recommendation list
     1456            JPanel actionsPanel = new JPanel(new FlowLayout(FlowLayout.LEFT)); //model selection buttons or configuration
     1457
     1458            addAndContinueButton = new JButton("Add and continue");
     1459            modelSettingsButton = new JButton("Model Settings");
     1460            useTagsCheckBox = new JCheckBox("Predict using tags");
     1461            recommendedClassesLabel = new JLabel("Recommended Classes:");
     1462
     1463            addAndContinueButton.addActionListener(new java.awt.event.ActionListener() {
     1464                @Override
     1465                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1466                    String selectedClass = categoryList.getSelectedValue();
     1467                    addAndContinueButtonActionPerformed(evt, selectedClass);
     1468
     1469                    //reconstruct vector for instance and use the model that was trained with classes here
     1470
     1471                    List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
     1472                    OsmPrimitive s;
     1473
     1474                    //get a simple selection
     1475                    if (!osmPrimitiveSelection.isEmpty()) {
     1476                        s = osmPrimitiveSelection.get(0);
     1477                        if (s.getInterestingTags().isEmpty()) {
     1478                            //load original model
     1479                            modelWithClasses = false;
     1480                            loadSVMmodel();
     1481                            createOSMObject(sel); //create object without class features
     1482                        } else {
     1483                            //recommend using tags: set the checkbox selected to avoid confusing the user
     1484                            useTagsCheckBox.setSelected(true);
     1485
     1486                            if (useTagsCheckBox.isSelected()) {
     1487                                //load model with classes
     1488                                modelWithClasses = true;
     1489                                loadSVMmodel();
     1490                                createOSMObject(sel); //create object including class features
     1491                            } else {
     1492                                modelWithClasses = false;
     1493                                loadSVMmodel();
     1494                                createOSMObject(sel); //create object including class features
     1495                            }
     1496                        }
     1497                    }
     1498                }
     1499            });
     1500
     1501            modelSettingsButton.addActionListener(new java.awt.event.ActionListener() {
     1502                @Override
     1503                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1504                    if (modelSettingsDialog == null) {
     1505                        System.out.println("new modelSettingsDialog");
     1506                        modelSettingsDialog = new ModelSettingsDialog(sel, addTagsDialog);
     1507                    } else {
     1508                        System.out.println("set modelSettingsDialog visible");
     1509                        modelSettingsDialog.makeVisible(true);
     1510                    }
     1511                }
     1512            });
     1513
     1514            useTagsCheckBox.addActionListener(new java.awt.event.ActionListener() {
     1515                @Override
     1516                public void actionPerformed(java.awt.event.ActionEvent evt) {
     1517                    List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
     1518                    OsmPrimitive s;
     1519                    if (!osmPrimitiveSelection.isEmpty()) {
     1520                        s = osmPrimitiveSelection.get(0);
     1521                        if (s.getInterestingTags().isEmpty()) {
     1522                            //load original model
     1523                            modelWithClasses = false;
     1524                            loadSVMmodel();
     1525                            createOSMObject(sel); //create object without class features
     1526                        } else {
     1527                            //useTagsCheckBox
     1528                            if (useTagsCheckBox.isSelected()) {
     1529                                //load model with classes
     1530                                modelWithClasses = true;
     1531                                loadSVMmodel();
     1532                                createOSMObject(sel); //create object including class features
     1533                            } else {
     1534                                modelWithClasses = false;
     1535                                loadSVMmodel();
     1536                                createOSMObject(sel); //create object including class features
     1537                            }
     1538                        }
     1539                    }
     1540                }
     1541            });
     1542
     1543            keys = new AutoCompletingComboBox();
     1544            values = new AutoCompletingComboBox();
     1545
     1546            mainPanel.add(new JLabel("<html>"+trn("This will change up to {0} object.",
     1547                    "This will change up to {0} objects.", sel.size(), sel.size())
     1548            +"<br><br>"+tr("Please select a key")), GBC.eol().fill(GBC.HORIZONTAL));
     1549
     1550            AutoCompletionManager autocomplete = MainApplication.getLayerManager().getEditLayer().data.getAutoCompletionManager();
     1551            List<AutoCompletionListItem> keyList = autocomplete.getKeys();
     1552
     1553            AutoCompletionListItem itemToSelect = null;
     1554            // remove the object's tag keys from the list
     1555            Iterator<AutoCompletionListItem> iter = keyList.iterator();
     1556            while (iter.hasNext()) {
     1557                AutoCompletionListItem item = iter.next();
     1558                if (item.getValue().equals(lastAddKey)) {
     1559                    itemToSelect = item;
     1560                }
     1561                for (int i = 0; i < tagData.getRowCount(); ++i) {
     1562                    if (item.getValue().equals(tagData.getValueAt(i, 0))) {
     1563                        if (itemToSelect == item) {
     1564                            itemToSelect = null;
     1565                        }
     1566                        iter.remove();
     1567                        break;
     1568                    }
     1569                }
     1570            }
     1571
     1572            Collections.sort(keyList, defaultACItemComparator);
     1573            keys.setPossibleACItems(keyList);
     1574            keys.setEditable(true);
     1575
     1576            mainPanel.add(keys, GBC.eop().fill());
     1577            mainPanel.add(new JLabel(tr("Please select a value")), GBC.eol());
     1578
     1579            model = new DefaultListModel<>();
     1580
     1581            parseTagsMappedToClasses();
     1582
     1583            List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
     1584            OsmPrimitive s;
     1585            //get a simple selection
     1586            if (!osmPrimitiveSelection.isEmpty()) {
     1587                s = osmPrimitiveSelection.get(0);
     1588                File modelDirectory = new File(MODEL_PATH);
     1589                String modelWithClassesPath = modelDirectory.getAbsolutePath() + "/model_with_classes";
     1590                File modelWithClassesFile = new File(modelWithClassesPath);
     1591                if (s.getInterestingTags().isEmpty() || !modelWithClassesFile.exists()) {
     1592                    modelWithClasses = false;
     1593                    loadSVMmodel(); //load original model
     1594                    createOSMObject(sel); //create object without class features
     1595                } else {
     1596                    //recommend using tags: set the checkbox selected to avoid confusing the user
     1597                    useTagsCheckBox.setSelected(true);
     1598                    modelWithClasses = true;
     1599                    loadSVMmodel(); //load model with classes
     1600                    createOSMObject(sel); //create object including class features
     1601                }
     1602            }
     1603
     1604            categoryList = new JList<>(model);
     1605
     1606            ListSelectionListener listSelectionListener = new ListSelectionListener() {
     1607                @Override
     1608                public void valueChanged(ListSelectionEvent listSelectionEvent) {
     1609                    if (!listSelectionEvent.getValueIsAdjusting()) { //This prevents double events
     1610
     1611                        String selectedClass = categoryList.getSelectedValue();
     1612
     1613                        if (selectedClass != null) { //null check, because the model is cleared after a new recommendation
     1614                            //tags become unselected
     1615                            if ((selectedClass.indexOf(" ")+1) > 0) {
     1616                                //add both key + value in tags
     1617                                String keyTag = selectedClass.substring(0, selectedClass.indexOf(" "));
     1618                                String valueTag = selectedClass.substring(selectedClass.indexOf(" ")+1, selectedClass.length());
     1619                                keys.setSelectedItem(keyTag); //adding selected tags to textBoxes
     1620                                values.setSelectedItem(valueTag);
     1621                            } else {
     1622                                //value does not have a value, add the key tag only
     1623                                String keyTag = selectedClass; //test it
     1624                                keys.setSelectedItem(keyTag);
     1625                                values.setSelectedItem("");
     1626                            }
     1627                        }
     1628                    }
     1629                }
     1630            };
     1631
     1632            categoryList.addListSelectionListener(listSelectionListener);
     1633            categoryList.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     1634            categoryList.setModel(model);
     1635
     1636            values.setEditable(true);
     1637            mainPanel.add(values, GBC.eop().fill());
     1638            if (itemToSelect != null) {
     1639                keys.setSelectedItem(itemToSelect);
     1640                if (lastAddValue != null) {
     1641                    values.setSelectedItem(lastAddValue);
     1642                }
     1643            }
     1644
     1645            FocusAdapter focus = addFocusAdapter(autocomplete, defaultACItemComparator);
     1646            // fire focus event in advance or otherwise the popup list will be too small at first
     1647            focus.focusGained(null);
     1648
     1649            int recentTagsToShow = PROPERTY_RECENT_TAGS_NUMBER.get();
     1650            if (recentTagsToShow > MAX_LRU_TAGS_NUMBER) {
     1651                recentTagsToShow = MAX_LRU_TAGS_NUMBER;
     1652            }
     1653
     1654            suggestRecentlyAddedTags(mainPanel, recentTagsToShow, focus);
     1655
     1656            mainPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     1657            listPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     1658            splitPanel.setBorder(BorderFactory.createEtchedBorder(EtchedBorder.LOWERED, Color.GRAY, Color.WHITE));
     1659
     1660            listPanel.add(recommendedClassesLabel, BorderLayout.NORTH);
     1661            listPanel.add(categoryList, BorderLayout.SOUTH);
     1662            actionsPanel.add(addAndContinueButton);
     1663            actionsPanel.add(modelSettingsButton);
     1664            actionsPanel.add(useTagsCheckBox);
     1665
     1666            recommendPanel.add(actionsPanel, BorderLayout.WEST);
     1667            recommendPanel.add(listPanel, BorderLayout.NORTH);
     1668
     1669            splitPanel.add(mainPanel, BorderLayout.WEST);
     1670            splitPanel.add(recommendPanel, BorderLayout.EAST);
     1671
     1672            setContent(splitPanel, false);
     1673
     1674            popupMenu.add(new AbstractAction(tr("Set number of recently added tags")) {
     1675                @Override
     1676                public void actionPerformed(ActionEvent e) {
     1677                    selectNumberOfTags();
     1678                }
     1679            });
     1680            JCheckBoxMenuItem rememberLastTags = new JCheckBoxMenuItem(
     1681                    new AbstractAction(tr("Remember last used tags after a restart")) {
     1682                        @Override
     1683                        public void actionPerformed(ActionEvent e) {
     1684                            boolean sel = ((JCheckBoxMenuItem) e.getSource()).getState();
     1685                            PROPERTY_REMEMBER_TAGS.put(sel);
     1686                            if (sel) saveTagsIfNeeded();
     1687                        }
     1688                    });
     1689            rememberLastTags.setState(PROPERTY_REMEMBER_TAGS.get());
     1690            popupMenu.add(rememberLastTags);
     1691        }
     1692
     1693        private void addAndContinueButtonActionPerformed(ActionEvent evt, String selectedClass) {
     1694            performTagAdding();
     1695        }
     1696
     1697        private void selectNumberOfTags() {
     1698            String s = JOptionPane.showInputDialog(this, tr("Please enter the number of recently added tags to display"));
     1699            if (s != null) try {
     1700                int v = Integer.parseInt(s);
     1701                if (v >= 0 && v <= MAX_LRU_TAGS_NUMBER) {
     1702                    PROPERTY_RECENT_TAGS_NUMBER.put(v);
     1703                    return;
     1704                }
     1705            } catch (NumberFormatException ex) {
     1706                Logging.warn(ex);
     1707            }
     1708            JOptionPane.showMessageDialog(this, tr("Please enter integer number between 0 and {0}", MAX_LRU_TAGS_NUMBER));
     1709        }
     1710
     1711        private void suggestRecentlyAddedTags(JPanel mainPanel, int tagsToShow, final FocusAdapter focus) {
     1712            if (!(tagsToShow > 0 && !recentTags.isEmpty()))
     1713                return;
     1714
     1715            mainPanel.add(new JLabel(tr("Recently added tags")), GBC.eol());
     1716
     1717            int count = 1;
     1718            // We store the maximum number (9) of recent tags to allow dynamic change of number of tags shown in the preferences.
     1719            // This implies to iterate in descending order,
     1720            // as the oldest elements will only be removed after we reach the maximum numbern and not the number of tags to show.
     1721            // However, as Set does not allow to iterate in descending order,
     1722            // we need to copy its elements into a List we can access in reverse order.
     1723            List<Tag> tags = new LinkedList<>(recentTags.keySet());
     1724            for (int i = tags.size()-1; i >= 0 && count <= tagsToShow; i--, count++) {
     1725                final Tag t = tags.get(i);
     1726                // Create action for reusing the tag, with keyboard shortcut Ctrl+(1-5)
     1727                String scKey = "properties:recent:"+count;
     1728                String scsKey = "properties:recent:shift:"+count;
     1729                Shortcut sc = Shortcut.registerShortcut(scKey, tr("Choose recent tag {0}", count), KeyEvent.VK_0+count, Shortcut.CTRL);
     1730                final JosmAction action = new JosmAction(scKey, null, tr("Use this tag again"), sc, false) {
     1731                    @Override
     1732                    public void actionPerformed(ActionEvent e) {
     1733                        keys.setSelectedItem(t.getKey());
     1734                        // fix #7951, #8298 - update list of values before setting value (?)
     1735                        focus.focusGained(null);
     1736                        values.setSelectedItem(t.getValue());
     1737                    }
     1738                };
     1739                Shortcut scs = Shortcut.registerShortcut(scsKey, tr("Apply recent tag {0}", count), KeyEvent.VK_0+count, Shortcut.CTRL_SHIFT);
     1740                final JosmAction actionShift = new JosmAction(scsKey, null, tr("Use this tag again"), scs, false) {
     1741                    @Override
     1742                    public void actionPerformed(ActionEvent e) {
     1743                        action.actionPerformed(null);
     1744                        performTagAdding();
     1745                    }
     1746                };
     1747                recentTagsActions.add(action);
     1748                recentTagsActions.add(actionShift);
     1749                disableTagIfNeeded(t, action);
     1750                // Find and display icon
     1751                ImageIcon icon = MapPaintStyles.getNodeIcon(t, false); // Filters deprecated icon
     1752                if (icon == null) {
     1753                    // If no icon found in map style look at presets
     1754                    Map<String, String> map = new HashMap<>();
     1755                    map.put(t.getKey(), t.getValue());
     1756                    // If still nothing display an empty icon
     1757                    if (icon == null) {
     1758                        icon = new ImageIcon(new BufferedImage(16, 16, BufferedImage.TYPE_INT_ARGB));
     1759                    }
     1760                }
     1761                GridBagConstraints gbc = new GridBagConstraints();
     1762                gbc.ipadx = 5;
     1763                mainPanel.add(new JLabel(action.isEnabled() ? icon : GuiHelper.getDisabledIcon(icon)), gbc);
     1764                // Create tag label
     1765                final String color = action.isEnabled() ? "" : "; color:gray";
     1766                final JLabel tagLabel = new JLabel("<html>"
     1767                        + "<style>td{border:1px solid gray; font-weight:normal"+color+"}</style>"
     1768                        + "<table><tr><td>" + XmlWriter.encode(t.toString(), true) + "</td></tr></table></html>");
     1769                if (action.isEnabled()) {
     1770                    // Register action
     1771                    mainPanel.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(sc.getKeyStroke(), scKey);
     1772                    mainPanel.getActionMap().put(scKey, action);
     1773                    mainPanel.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(scs.getKeyStroke(), scsKey);
     1774                    mainPanel.getActionMap().put(scsKey, actionShift);
     1775                    // Make the tag label clickable and set tooltip to the action description (this displays also the keyboard shortcut)
     1776                    tagLabel.setToolTipText((String) action.getValue(Action.SHORT_DESCRIPTION));
     1777                    tagLabel.setCursor(Cursor.getPredefinedCursor(Cursor.HAND_CURSOR));
     1778                    tagLabel.addMouseListener(new MouseAdapter() {
     1779                        @Override
     1780                        public void mouseClicked(MouseEvent e) {
     1781                            action.actionPerformed(null);
     1782                            // add tags and close window on double-click
     1783
     1784                            if (e.getClickCount() > 1) {
     1785                                buttonAction(0, null); // emulate OK click and close the dialog
     1786                            }
     1787                            // add tags on Shift-Click
     1788                            if (e.isShiftDown()) {
     1789                                performTagAdding();
     1790                            }
     1791                        }
     1792                    });
     1793                } else {
     1794                    // Disable tag label
     1795                    tagLabel.setEnabled(false);
     1796                    // Explain in the tooltip why
     1797                    tagLabel.setToolTipText(tr("The key ''{0}'' is already used", t.getKey()));
     1798                }
     1799                // Finally add label to the resulting panel
     1800                JPanel tagPanel = new JPanel(new FlowLayout(FlowLayout.LEFT, 0, 0));
     1801                tagPanel.add(tagLabel);
     1802                mainPanel.add(tagPanel, GBC.eol().fill(GBC.HORIZONTAL));
     1803            }
     1804        }
     1805
     1806        public void destroyActions() {
     1807            for (JosmAction action : recentTagsActions) {
     1808                action.destroy();
     1809            }
     1810        }
     1811
     1812        /**
     1813        * Read tags from comboboxes and add it to all selected objects
     1814        */
     1815        public final void performTagAdding() {
     1816            String key = Tag.removeWhiteSpaces(keys.getEditor().getItem().toString());
     1817            String value = Tag.removeWhiteSpaces(values.getEditor().getItem().toString());
     1818            if (key.isEmpty() || value.isEmpty()) return;
     1819            for (OsmPrimitive osm: sel) {
     1820                String val = osm.get(key);
     1821                if (val != null && !val.equals(value)) {
     1822                    if (!warnOverwriteKey(tr("You changed the value of ''{0}'' from ''{1}'' to ''{2}''.", key, val, value),
     1823                            "overwriteAddKey"))
     1824                        return;
     1825                    break;
     1826                }
     1827            }
     1828            lastAddKey = key;
     1829            lastAddValue = value;
     1830            recentTags.put(new Tag(key, value), null);
     1831            AutoCompletionManager.rememberUserInput(key, value, false);
     1832            commandCount++;
     1833            MainApplication.undoRedo.add(new ChangePropertyCommand(sel, key, value));
     1834            changedKey = key;
     1835        }
     1836
     1837        public void undoAllTagsAdding() {
     1838            MainApplication.undoRedo.undo(commandCount);
     1839        }
     1840
     1841        private void disableTagIfNeeded(final Tag t, final JosmAction action) {
     1842            // Disable action if its key is already set on the object (the key being absent from the keys list for this reason
     1843            // performing this action leads to autocomplete to the next key (see #7671 comments)
     1844            for (int j = 0; j < tagData.getRowCount(); ++j) {
     1845                if (t.getKey().equals(tagData.getValueAt(j, 0))) {
     1846                    action.setEnabled(false);
     1847                    break;
     1848                }
     1849            }
     1850        }
     1851
     1852        private void loadSVMmodel() {
     1853            File modelDirectory = new File(MODEL_PATH);
     1854            File modelFile;
     1855            if (useCombinedModel) {
     1856                if (filesAndWeights.isEmpty()) {
     1857                    System.out.println("No models selected! Loading defaults..");
     1858                    if (modelWithClasses) {
     1859                        System.out.println("Using default/last model with classes: " + modelDirectory.getAbsolutePath() + "/model_with_classes");
     1860                        modelFile = new File(modelDirectory.getAbsolutePath() + "/model_with_classes");
     1861                        try {
     1862                            System.out.println("try to load model: " + modelFile.getAbsolutePath());
     1863                            modelSVM = Model.load(modelFile);
     1864                            System.out.println("model loaded!");
     1865
     1866                        } catch (IOException ex) {
     1867                            Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
     1868                        }
     1869                        modelSVMLabelSize = modelSVM.getLabels().length;
     1870                        modelSVMLabels = modelSVM.getLabels();
     1871                    } else {
     1872                        System.out.println("Using default/last model without classes: " + modelDirectory.getAbsolutePath() + "/best_model");
     1873                        modelFile = new File(modelDirectory.getAbsolutePath() + "/best_model");
     1874                        try {
     1875                            System.out.println("try to load model: " + modelFile.getAbsolutePath());
     1876                            modelSVM = Model.load(modelFile);
     1877                            System.out.println("model loaded!");
     1878
     1879                        } catch (IOException ex) {
     1880                            Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
     1881                        }
     1882                        modelSVMLabelSize = modelSVM.getLabels().length;
     1883                        modelSVMLabels = modelSVM.getLabels();
     1884                    }
     1885                }
     1886                if (modelWithClasses) { //check filenames to define if model with classes is selected
     1887                    System.out.println("Using combined model with classes");
     1888                    useCombinedSVMmodels(sel, true);
     1889                } else {
     1890                    System.out.println("Using combined model without classes");
     1891                    useCombinedSVMmodels(sel, false);
     1892                }
     1893            } else if (useCustomSVMModel) {
     1894                System.out.println("custom path: " + customSVMModelPath);
     1895                File checkExistance = new File(customSVMModelPath);
     1896                if (checkExistance.exists() && checkExistance.isFile()) {
     1897                    if (modelWithClasses) {
     1898                        System.out.println("Using custom model with classes: ");
     1899                        if (customSVMModelPath.endsWith(".0")) {
     1900                            String customSVMModelPathWithClasses = customSVMModelPath.substring(0, customSVMModelPath.length() - 2) + ".1";
     1901
     1902                            modelFile = new File(customSVMModelPathWithClasses);
     1903                            System.out.println(customSVMModelPathWithClasses);
     1904                        } else {
     1905                            modelFile = new File(customSVMModelPath);
     1906                        }
     1907                    } else {
     1908                        System.out.println("Using custom model without classes");
     1909                        if (customSVMModelPath.endsWith(".1")) {
     1910                            String customSVMModelPathWithoutClasses = customSVMModelPath.substring(0, customSVMModelPath.length() - 2) + ".0";
     1911                            modelFile = new File(customSVMModelPathWithoutClasses);
     1912                            System.out.println(customSVMModelPathWithoutClasses);
     1913                        } else {
     1914                            modelFile = new File(customSVMModelPath);
     1915                        }
     1916                    }
     1917                    try {
     1918                        System.out.println("try to load model: " + modelFile.getAbsolutePath());
     1919                        modelSVM = Model.load(modelFile);
     1920                        System.out.println("model loaded!");
     1921
     1922                    } catch (IOException ex) {
     1923                        Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
     1924                    }
     1925                    modelSVMLabelSize = modelSVM.getLabels().length;
     1926                    modelSVMLabels = modelSVM.getLabels();
     1927
     1928                } else {
     1929                    //user chose to use a custom model, but did not provide a path to a model:
     1930                    if (modelWithClasses) {
     1931                        System.out.println("Using default/last model with classes");
     1932                        modelFile = new File(modelDirectory.getAbsolutePath() + "/model_with_classes");
     1933                    } else {
     1934                        System.out.println("Using default/last model without classes");
     1935                        modelFile = new File(modelDirectory.getAbsolutePath() + "/best_model");
     1936                    }
     1937
     1938                    try {
     1939                        System.out.println("try to load model: " + modelFile.getAbsolutePath());
     1940                        modelSVM = Model.load(modelFile);
     1941                        System.out.println("model loaded!");
     1942
     1943                    } catch (IOException ex) {
     1944                        Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
     1945                    }
     1946                    modelSVMLabelSize = modelSVM.getLabels().length;
     1947                    modelSVMLabels = modelSVM.getLabels();
     1948
     1949                }
     1950            } else {
     1951                if (modelWithClasses) {
     1952                    System.out.println("Using default/last model with classes");
     1953                    modelFile = new File(modelDirectory.getAbsolutePath() + "/model_with_classes");
     1954                } else {
     1955                    System.out.println("Using default/last model without classes");
     1956                    modelFile = new File(modelDirectory.getAbsolutePath() + "/best_model");
     1957                }
     1958
     1959                try {
     1960                    System.out.println("try to load model: " + modelFile.getAbsolutePath());
     1961                    modelSVM = Model.load(modelFile);
     1962                    System.out.println("model loaded!");
     1963
     1964                } catch (IOException ex) {
     1965                    Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
     1966                }
     1967                modelSVMLabelSize = modelSVM.getLabels().length;
     1968                modelSVMLabels = modelSVM.getLabels();
     1969            }
     1970        }
     1971
     1972        private void useCombinedSVMmodels(Collection<OsmPrimitive> sel, boolean useClassFeatures) {
     1973            System.out.println("The system will combine " + filesAndWeights.size() + " SVM models.");
     1974
     1975            MathTransform transform = null;
     1976            GeometryFactory geometryFactory = new GeometryFactory();
     1977            CoordinateReferenceSystem sourceCRS = DefaultGeographicCRS.WGS84;
     1978            CoordinateReferenceSystem targetCRS = DefaultGeocentricCRS.CARTESIAN;
     1979            try {
     1980                transform = CRS.findMathTransform(sourceCRS, targetCRS, true);
     1981
     1982            } catch (FactoryException ex) {
     1983                Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     1984            }
     1985
     1986            OSMWay selectedInstance;
     1987            List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
     1988            OsmPrimitive s;
     1989
     1990            //get a simple selection
     1991            if (!osmPrimitiveSelection.isEmpty()) {
     1992                s = osmPrimitiveSelection.get(0);
     1993            } else {
     1994                return;
     1995            }
     1996
     1997            selectedInstance = new OSMWay();
     1998            for (Way selectedWay : s.getDataSet().getSelectedWays()) {
     1999                List<Node> selectedWayNodes = selectedWay.getNodes();
     2000                for (Node node : selectedWayNodes) {
     2001                    node.getCoor();
     2002                    if (node.isLatLonKnown()) {
     2003                        double lat = node.getCoor().lat();
     2004                        double lon = node.getCoor().lon();
     2005
     2006                        Coordinate sourceCoordinate = new Coordinate(lon, lat);
     2007                        Coordinate targetGeometry = null;
     2008                        try {
     2009                            targetGeometry = JTS.transform(sourceCoordinate, null, transform);
     2010                        } catch (MismatchedDimensionException | TransformException ex) {
     2011                            Logger.getLogger(OSMParser.class.getName()).log(Level.SEVERE, null, ex);
     2012                        }
     2013
     2014                        Geometry geom = geometryFactory.createPoint(new Coordinate(targetGeometry));
     2015                        selectedInstance.addNodeGeometry(geom);
     2016                    }
     2017                }
     2018            }
     2019            Geometry fullGeom = geometryFactory.buildGeometry(selectedInstance.getNodeGeometries());
     2020            if ((selectedInstance.getNodeGeometries().size() > 3) &&
     2021                    selectedInstance.getNodeGeometries().get(0).equals(selectedInstance.getNodeGeometries()
     2022                            .get(selectedInstance.getNodeGeometries().size()-1))) {
     2023                //checks if the beginning and ending node are the same and the number of nodes are more than 3.
     2024                //the nodes must be more than 3, because jts does not allow a construction of a linear ring with less points.
     2025                LinearRing linear = geometryFactory.createLinearRing(fullGeom.getCoordinates());
     2026                Polygon poly = new Polygon(linear, null, geometryFactory);
     2027                selectedInstance.setGeometry(poly);
     2028
     2029                System.out.println("\n\npolygon");
     2030            } else if (selectedInstance.getNodeGeometries().size() > 1) {
     2031                //it is an open geometry with more than one nodes, make it linestring
     2032                System.out.println("\n\nlinestring");
     2033                LineString lineString = geometryFactory.createLineString(fullGeom.getCoordinates());
     2034                selectedInstance.setGeometry(lineString);
     2035            } else { //we assume all the rest geometries are points
     2036                System.out.println("\n\npoint");
     2037                Point point = geometryFactory.createPoint(fullGeom.getCoordinate());
     2038                selectedInstance.setGeometry(point);
     2039            }
     2040
     2041            Map<String, String> selectedTags = s.getInterestingTags();
     2042            selectedInstance.setAllTags(selectedTags);
     2043
     2044            //construct vector
     2045            if (selectedInstance != null) {
     2046                int id;
     2047
     2048                OSMClassification classifier = new OSMClassification();
     2049                classifier.calculateClasses(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
     2050
     2051                if (useClassFeatures) {
     2052                    ClassFeatures classFeatures = new ClassFeatures();
     2053                    classFeatures.createClassFeatures(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
     2054                    id = 1422;
     2055                } else {
     2056                    id = 1;
     2057                }
     2058
     2059                GeometryFeatures geometryFeatures = new GeometryFeatures(id);
     2060                geometryFeatures.createGeometryFeatures(selectedInstance);
     2061                id = geometryFeatures.getLastID();
     2062                TextualFeatures textualFeatures = new TextualFeatures(id, textualList, languageDetector);
     2063                textualFeatures.createTextualFeatures(selectedInstance);
     2064
     2065                List<FeatureNode> featureNodeList = selectedInstance.getFeatureNodeList();
     2066
     2067                FeatureNode[] featureNodeArray = new FeatureNode[featureNodeList.size()];
     2068
     2069                int i = 0;
     2070                for (FeatureNode featureNode : featureNodeList) {
     2071                    featureNodeArray[i] = featureNode;
     2072                    i++;
     2073                }
     2074                FeatureNode[] testInstance2 = featureNodeArray;
     2075
     2076                //compute prediction list for every model
     2077                int[] ranks = new int[10];
     2078
     2079                for (int l = 0; l < 10; l++) {
     2080                    ranks[l] = 10-l; //init from 10 to 1
     2081                }
     2082
     2083                Map<String, Double> scoreMap = new HashMap<>();
     2084
     2085                Map<File, Double> alignedFilesAndWeights = getAlignedModels(filesAndWeights);
     2086
     2087                for (File modelFile : alignedFilesAndWeights.keySet()) {
     2088
     2089                    try {
     2090                        modelSVM = Model.load(modelFile);
     2091                    } catch (IOException ex) {
     2092                        Logger.getLogger(TrainWorker.class.getName()).log(Level.SEVERE, null, ex);
     2093                    }
     2094                    modelSVMLabelSize = modelSVM.getLabels().length;
     2095                    modelSVMLabels = modelSVM.getLabels();
     2096
     2097                    Map<Integer, Integer> mapLabelsToIDs = new HashMap<>();
     2098                    for (int h = 0; h < modelSVMLabelSize; h++) {
     2099                        mapLabelsToIDs.put(modelSVMLabels[h], h);
     2100                    }
     2101                    double[] scores = new double[modelSVMLabelSize];
     2102                    Linear.predictValues(modelSVM, testInstance2, scores);
     2103
     2104                    Map<Double, Integer> scoresValues = new HashMap<>();
     2105                    for (int h = 0; h < scores.length; h++) {
     2106                        scoresValues.put(scores[h], h);
     2107                    }
     2108
     2109                    Arrays.sort(scores);
     2110                    int predicted1 = modelSVMLabels[scoresValues.get(scores[scores.length-1])];
     2111                    int predicted2 = modelSVMLabels[scoresValues.get(scores[scores.length-2])];
     2112                    int predicted3 = modelSVMLabels[scoresValues.get(scores[scores.length-3])];
     2113                    int predicted4 = modelSVMLabels[scoresValues.get(scores[scores.length-4])];
     2114                    int predicted5 = modelSVMLabels[scoresValues.get(scores[scores.length-5])];
     2115                    int predicted6 = modelSVMLabels[scoresValues.get(scores[scores.length-6])];
     2116                    int predicted7 = modelSVMLabels[scoresValues.get(scores[scores.length-7])];
     2117                    int predicted8 = modelSVMLabels[scoresValues.get(scores[scores.length-8])];
     2118                    int predicted9 = modelSVMLabels[scoresValues.get(scores[scores.length-9])];
     2119                    int predicted10 = modelSVMLabels[scoresValues.get(scores[scores.length-10])];
     2120
     2121                    String[] predictedTags = new String[10];
     2122                    for (Map.Entry<String, Integer> entry : mapperWithIDs.entrySet()) {
     2123
     2124                        if (entry.getValue().equals(predicted1)) {
     2125                            predictedTags[0] = entry.getKey();
     2126                        } else if (entry.getValue().equals(predicted2)) {
     2127                            predictedTags[1] = entry.getKey();
     2128                        } else if (entry.getValue().equals(predicted3)) {
     2129                            predictedTags[2] = entry.getKey();
     2130                        } else if (entry.getValue().equals(predicted4)) {
     2131                            predictedTags[3] = entry.getKey();
     2132                        } else if (entry.getValue().equals(predicted5)) {
     2133                            predictedTags[4] = entry.getKey();
     2134                        } else if (entry.getValue().equals(predicted6)) {
     2135                            predictedTags[5] = entry.getKey();
     2136                        } else if (entry.getValue().equals(predicted7)) {
     2137                            predictedTags[6] = entry.getKey();
     2138                        } else if (entry.getValue().equals(predicted8)) {
     2139                            predictedTags[7] = entry.getKey();
     2140                        } else if (entry.getValue().equals(predicted9)) {
     2141                            predictedTags[8] = entry.getKey();
     2142                        } else if (entry.getValue().equals(predicted10)) {
     2143                            predictedTags[9] = entry.getKey();
     2144                        }
     2145                    }
     2146                    //clearing model, to add the new computed classes in jlist
     2147                    model.clear();
     2148                    for (Map.Entry<String, String> tag : mappings.entrySet()) {
     2149
     2150                        for (int k = 0; k < 10; k++) {
     2151                            if (tag.getValue().equals(predictedTags[k])) {
     2152                                predictedTags[k] = tag.getKey();
     2153                                model.addElement(tag.getKey());
     2154                            }
     2155                        }
     2156                    }
     2157                    System.out.println("combined, predicted classes: " + Arrays.toString(predictedTags));
     2158
     2159                    for (int r = 0; r < ranks.length; r++) {
     2160                        String predictedTag = predictedTags[r];
     2161                        Double currentWeight = alignedFilesAndWeights.get(modelFile);
     2162                        double finalRank = ranks[r]*currentWeight;
     2163
     2164                        if (scoreMap.containsKey(predictedTag)) {
     2165                            Double scoreToAdd = scoreMap.get(predictedTag);
     2166                            scoreMap.put(predictedTag, finalRank+scoreToAdd);
     2167                        } else {
     2168                            scoreMap.put(predictedTag, finalRank);
     2169                        }
     2170                        //add final weight - predicted tag
     2171                    }
     2172                } //files iter
     2173                model.clear();
     2174                List<Double> scoresList = new ArrayList<>(scoreMap.values());
     2175                Collections.sort(scoresList, Collections.reverseOrder());
     2176
     2177                for (Double sco : scoresList) {
     2178                    if (model.size() > 9) {
     2179                        break;
     2180                    }
     2181                    for (Map.Entry<String, Double> scoreEntry : scoreMap.entrySet()) {
     2182                        if (scoreEntry.getValue().equals(sco)) {
     2183                            model.addElement(scoreEntry.getKey());
     2184                        }
     2185                    }
     2186                }
     2187            }
     2188        }
     2189
     2190        private void createOSMObject(Collection<OsmPrimitive> sel) {
     2191
     2192            MathTransform transform = null;
     2193            GeometryFactory geometryFactory = new GeometryFactory();
     2194            CoordinateReferenceSystem sourceCRS = DefaultGeographicCRS.WGS84;
     2195            CoordinateReferenceSystem targetCRS = DefaultGeocentricCRS.CARTESIAN;
     2196            try {
     2197                transform = CRS.findMathTransform(sourceCRS, targetCRS, true);
     2198
     2199            } catch (FactoryException ex) {
     2200                Logger.getLogger(OSMRecPluginHelper.class.getName()).log(Level.SEVERE, null, ex);
     2201            }
     2202
     2203            //fire an error to the user if he has multiple selection from map
     2204
     2205            //we consider simple (one instance) selection, so we get the first of the sel list
     2206
     2207            OSMWay selectedInstance;
     2208            List<OsmPrimitive> osmPrimitiveSelection = new ArrayList<>(sel);
     2209            OsmPrimitive s;
     2210
     2211            //get a simple selection
     2212            if (!osmPrimitiveSelection.isEmpty()) {
     2213                s = osmPrimitiveSelection.get(0);
     2214            } else {
     2215                return;
     2216            }
     2217
     2218            selectedInstance = new OSMWay();
     2219            for (Way selectedWay : s.getDataSet().getSelectedWays()) {
     2220                List<Node> selectedWayNodes = selectedWay.getNodes();
     2221                for (Node node : selectedWayNodes) {
     2222                    node.getCoor();
     2223                    if (node.isLatLonKnown()) {
     2224                        double lat = node.getCoor().lat();
     2225                        double lon = node.getCoor().lon();
     2226
     2227                        Coordinate sourceCoordinate = new Coordinate(lon, lat);
     2228                        Coordinate targetGeometry = null;
     2229                        try {
     2230                            targetGeometry = JTS.transform(sourceCoordinate, null, transform);
     2231                        } catch (MismatchedDimensionException | TransformException ex) {
     2232                            Logger.getLogger(OSMParser.class.getName()).log(Level.SEVERE, null, ex);
     2233                        }
     2234
     2235                        Geometry geom = geometryFactory.createPoint(new Coordinate(targetGeometry));
     2236                        selectedInstance.addNodeGeometry(geom);
     2237                    }
     2238                }
     2239            }
     2240            Geometry fullGeom = geometryFactory.buildGeometry(selectedInstance.getNodeGeometries());
     2241
     2242            System.out.println("number of nodes: " + selectedInstance.getNodeGeometries().size());
     2243
     2244            if ((selectedInstance.getNodeGeometries().size() > 3) &&
     2245                    selectedInstance.getNodeGeometries().get(0).equals(selectedInstance.getNodeGeometries()
     2246                            .get(selectedInstance.getNodeGeometries().size()-1))) {
     2247                //checks if the beginning and ending node are the same and the number of nodes are more than 3.
     2248                //the nodes must be more than 3, because jts does not allow a construction of a linear ring with less points.
     2249                LinearRing linear = geometryFactory.createLinearRing(fullGeom.getCoordinates());
     2250                Polygon poly = new Polygon(linear, null, geometryFactory);
     2251                selectedInstance.setGeometry(poly);
     2252
     2253                System.out.println("\n\npolygon");
     2254            } else if (selectedInstance.getNodeGeometries().size() > 1) {
     2255                //it is an open geometry with more than one nodes, make it linestring
     2256                System.out.println("\n\nlinestring");
     2257                LineString lineString = geometryFactory.createLineString(fullGeom.getCoordinates());
     2258                selectedInstance.setGeometry(lineString);
     2259            } else { //we assume all the rest geometries are points
     2260                System.out.println("\n\npoint");
     2261                Point point = geometryFactory.createPoint(fullGeom.getCoordinate());
     2262                selectedInstance.setGeometry(point);
     2263            }
     2264
     2265            Map<String, String> selectedTags = s.getInterestingTags();
     2266            selectedInstance.setAllTags(selectedTags);
     2267
     2268            //construct vector here
     2269            if (selectedInstance != null) {
     2270                int id;
     2271                if (mappings == null) {
     2272                    System.out.println("null mappings ERROR");
     2273                }
     2274
     2275                OSMClassification classifier = new OSMClassification();
     2276                classifier.calculateClasses(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
     2277
     2278                if (modelWithClasses) {
     2279                    ClassFeatures classFeatures = new ClassFeatures();
     2280                    classFeatures.createClassFeatures(selectedInstance, mappings, mapperWithIDs, indirectClasses, indirectClassesWithIDs);
     2281                    id = 1422;
     2282                } else {
     2283                    id = 1;
     2284                }
     2285
     2286                GeometryFeatures geometryFeatures = new GeometryFeatures(id);
     2287                geometryFeatures.createGeometryFeatures(selectedInstance);
     2288                id = geometryFeatures.getLastID();
     2289                TextualFeatures textualFeatures = new TextualFeatures(id, textualList, languageDetector);
     2290                textualFeatures.createTextualFeatures(selectedInstance);
     2291
     2292                List<FeatureNode> featureNodeList = selectedInstance.getFeatureNodeList();
     2293                System.out.println(featureNodeList);
     2294
     2295                FeatureNode[] featureNodeArray = new FeatureNode[featureNodeList.size()];
     2296
     2297                int i = 0;
     2298                for (FeatureNode featureNode : featureNodeList) {
     2299                    featureNodeArray[i] = featureNode;
     2300                    i++;
     2301                }
     2302                FeatureNode[] testInstance2 = featureNodeArray;
     2303
     2304                Map<Integer, Integer> mapLabelsToIDs = new HashMap<>();
     2305                for (int h = 0; h < modelSVMLabelSize; h++) {
     2306                    mapLabelsToIDs.put(modelSVMLabels[h], h);
     2307                }
     2308
     2309                double[] scores = new double[modelSVMLabelSize];
     2310                Linear.predictValues(modelSVM, testInstance2, scores);
     2311
     2312                Map<Double, Integer> scoresValues = new HashMap<>();
     2313                for (int h = 0; h < scores.length; h++) {
     2314                    scoresValues.put(scores[h], h);
     2315                }
     2316
     2317                Arrays.sort(scores);
     2318
     2319                int[] preds = new int[RECOMMENDATIONS_SIZE];
     2320                for (int p = 0; p < RECOMMENDATIONS_SIZE; p++) {
     2321                    preds[p] = modelSVMLabels[scoresValues.get(scores[scores.length-(p+1)])];
     2322                }
     2323                String[] predictedTags2 = new String[RECOMMENDATIONS_SIZE];
     2324
     2325                for (int p = 0; p < RECOMMENDATIONS_SIZE; p++) {
     2326                    if (idsWithMappings.containsKey(preds[p])) {
     2327                        predictedTags2[p] = idsWithMappings.get(preds[p]);
     2328                    }
     2329                }
     2330
     2331                //clearing model, to add the new computed classes in jlist
     2332                model.clear();
     2333                for (Map.Entry<String, String> tag : mappings.entrySet()) {
     2334                    for (int k = 0; k < 10; k++) {
     2335                        if (tag.getValue().equals(predictedTags2[k])) {
     2336                            predictedTags2[k] = tag.getKey();
     2337                            model.addElement(tag.getKey());
     2338                        }
     2339                    }
     2340                }
     2341                System.out.println("Optimized - create OSMObject, predicted classes: " + Arrays.toString(predictedTags2));
     2342            }
     2343        }
     2344
     2345        private void parseTagsMappedToClasses() {
     2346
     2347            InputStream tagsToClassesMapping = TrainWorker.class.getResourceAsStream("/resources/files/Map");
     2348            Mapper mapper = new Mapper();
     2349            try {
     2350                mapper.parseFile(tagsToClassesMapping);
     2351
     2352            } catch (FileNotFoundException ex) {
     2353                Logger.getLogger(Mapper.class.getName()).log(Level.SEVERE, null, ex);
     2354            }
     2355            mappings = mapper.getMappings();
     2356            mapperWithIDs = mapper.getMappingsWithIDs();
     2357            idsWithMappings = mapper.getIDsWithMappings();
     2358        }
     2359
     2360        private void loadTextualList(File textualListFile) {
     2361
     2362            Scanner input = null;
     2363
     2364            try {
     2365                input = new Scanner(textualListFile);
     2366            } catch (FileNotFoundException ex) {
     2367                Logging.warn(ex);
     2368            }
     2369            while (input.hasNext()) {
     2370                String nextLine = input.nextLine();
     2371                textualList.add(nextLine);
     2372            }
     2373            System.out.println("Textual List parsed from file successfully." + textualList);
     2374        }
     2375
     2376        private void loadDefaultTextualList() {
     2377
     2378            InputStream textualListStream = TrainWorker.class.getResourceAsStream("/resources/files/textualList.txt");
     2379            TextualStatistics textualStatistics = new TextualStatistics();
     2380            textualStatistics.parseTextualList(textualListStream);
     2381            textualList = textualStatistics.getTextualList();
     2382            System.out.println("Default Textual List parsed from file successfully." + textualList);
     2383        }
     2384
     2385        private void loadOntology() {
     2386            InputStream ontologyStream = TrainWorker.class.getResourceAsStream("/resources/files/owl.xml");
     2387            Ontology ontology = new Ontology(ontologyStream);
     2388            indirectClasses = ontology.getIndirectClasses();
     2389            indirectClassesWithIDs = ontology.getIndirectClassesIDs();
     2390        }
     2391
     2392        private Map<File, Double> getAlignedModels(Map<File, Double> filesAndWeights) {
     2393            Map<File, Double> alignedFilesAndWeights = new HashMap<>();
     2394            if (modelWithClasses) {
     2395                for (Entry<File, Double> entry : filesAndWeights.entrySet()) {
     2396                    String absolutePath = entry.getKey().getAbsolutePath();
     2397                    if (absolutePath.endsWith(".0")) {
     2398                        String newPath = absolutePath.substring(0, absolutePath.length()-2) + ".1";
     2399                        File alignedFile = new File(newPath);
     2400                        if (alignedFile.exists()) {
     2401                            alignedFilesAndWeights.put(alignedFile, entry.getValue());
     2402                        }
     2403                    } else {
     2404                        alignedFilesAndWeights.put(entry.getKey(), entry.getValue());
     2405                    }
     2406                }
     2407            } else {
     2408                for (Entry<File, Double> entry : filesAndWeights.entrySet()) {
     2409                    String absolutePath = entry.getKey().getAbsolutePath();
     2410                    if (absolutePath.endsWith(".1")) {
     2411                        String newPath = absolutePath.substring(0, absolutePath.length()-2) + ".0";
     2412                        File alignedFile = new File(newPath);
     2413                        if (alignedFile.exists()) {
     2414                            alignedFilesAndWeights.put(alignedFile, entry.getValue());
     2415                        }
     2416                    } else {
     2417                        alignedFilesAndWeights.put(entry.getKey(), entry.getValue());
     2418                    }
     2419                }
     2420            }
     2421            return alignedFilesAndWeights;
     2422        }
     2423    }
    24242424}
Note: See TracChangeset for help on using the changeset viewer.