JAL-591 - refactored web service parameter dialog to allow options and parameters...
[jalview.git] / src / jalview / gui / WsJobParameters.java
1 /*
2  * Jalview - A Sequence Alignment Editor and Viewer (Version 2.6)
3  * Copyright (C) 2010 J Procter, AM Waterhouse, G Barton, M Clamp, S Searle
4  * 
5  * This file is part of Jalview.
6  * 
7  * Jalview is free software: you can redistribute it and/or
8  * modify it under the terms of the GNU General Public License 
9  * as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
10  * 
11  * Jalview is distributed in the hope that it will be useful, but 
12  * WITHOUT ANY WARRANTY; without even the implied warranty 
13  * of MERCHANTABILITY or FITNESS FOR A PARTICULAR 
14  * PURPOSE.  See the GNU General Public License for more details.
15  * 
16  * You should have received a copy of the GNU General Public License along with Jalview.  If not, see <http://www.gnu.org/licenses/>.
17  */
18 package jalview.gui;
19
20 import jalview.gui.OptsAndParamsPage.OptionBox;
21 import jalview.gui.OptsAndParamsPage.ParamBox;
22 import jalview.ws.jws2.JabaParamStore;
23 import jalview.ws.jws2.JabaPreset;
24 import jalview.ws.jws2.Jws2Discoverer;
25 import jalview.ws.jws2.Jws2Discoverer.Jws2Instance;
26 import jalview.ws.params.ArgumentI;
27 import jalview.ws.params.OptionI;
28 import jalview.ws.params.ParamDatastoreI;
29 import jalview.ws.params.ParameterI;
30 import jalview.ws.params.ValueConstrainI;
31 import jalview.ws.params.WsParamSetI;
32
33 import java.awt.BorderLayout;
34 import java.awt.Component;
35 import java.awt.Container;
36 import java.awt.Dimension;
37 import java.awt.FlowLayout;
38 import java.awt.Font;
39 import java.awt.GridBagConstraints;
40 import java.awt.GridBagLayout;
41 import java.awt.GridLayout;
42 import java.awt.Rectangle;
43 import java.awt.event.ActionEvent;
44 import java.awt.event.ActionListener;
45 import java.awt.event.ItemEvent;
46 import java.awt.event.ItemListener;
47 import java.awt.event.MouseEvent;
48 import java.awt.event.MouseListener;
49 import java.awt.event.WindowEvent;
50 import java.awt.event.WindowListener;
51 import java.net.URL;
52 import java.util.ArrayList;
53 import java.util.Hashtable;
54 import java.util.Iterator;
55 import java.util.List;
56 import java.util.Map;
57 import java.util.Vector;
58
59 import javax.swing.JButton;
60 import javax.swing.JCheckBox;
61 import javax.swing.JComboBox;
62 import javax.swing.JComponent;
63 import javax.swing.JDialog;
64 import javax.swing.JFrame;
65 import javax.swing.JLabel;
66 import javax.swing.JMenuItem;
67 import javax.swing.JOptionPane;
68 import javax.swing.JPanel;
69 import javax.swing.JPopupMenu;
70 import javax.swing.JScrollPane;
71 import javax.swing.JSlider;
72 import javax.swing.JSplitPane;
73 import javax.swing.JTabbedPane;
74 import javax.swing.JTextArea;
75 import javax.swing.JTextField;
76 import javax.swing.border.TitledBorder;
77 import javax.swing.event.ChangeEvent;
78 import javax.swing.event.ChangeListener;
79 import javax.swing.event.DocumentEvent;
80 import javax.swing.event.DocumentListener;
81
82 import compbio.metadata.Argument;
83 import compbio.metadata.Option;
84 import compbio.metadata.Parameter;
85 import compbio.metadata.Preset;
86 import compbio.metadata.PresetManager;
87 import compbio.metadata.RunnerConfig;
88 import compbio.metadata.ValueConstrain.Type;
89
90 /**
91  * job parameter editing/browsing dialog box. User can browse existing settings
92  * (user + presets + Defaults), and any changes to parameters creates a modified
93  * user parameter set. LOGIC: If the parameter set is modified, and its name is
94  * a valid, non-existant user parameter set, then a save button is shown. If the
95  * parameter set is modified and its name is a valid, extant user parameter set,
96  * then an update button is shown. If user parameter set's name is edited, and
97  * old name exists as a writable user parameter set, then rename button is
98  * shown. If current parameter set is associated with a user defined parameter
99  * set, then : if set is modifed, a 'revert' button is shown. if set is not
100  * modified, a 'delete' button is shown.
101  * 
102  * @author JimP
103  * 
104  */
105 public class WsJobParameters extends JPanel implements ItemListener,
106         ActionListener, DocumentListener, OptsParametersContainerI
107 {
108   URL linkImageURL = getClass().getResource("/images/link.gif");
109
110   private static final String SVC_DEF = "Defaults"; // this is the null
111                                                     // parameter set as shown to
112                                                     // user
113
114   /**
115    * manager for options and parameters.
116    */
117   OptsAndParamsPage opanp = new OptsAndParamsPage(this);
118
119   /**
120    * panel containing job options
121    */
122   JPanel jobOptions = new JPanel();
123
124   /**
125    * panel containing job parameters
126    */
127   JPanel paramList = new JPanel();
128
129   
130   
131   JPanel SetNamePanel = new JPanel();
132
133   JPanel setDetails = new JPanel();
134
135   JSplitPane settingsPanel = new JSplitPane();
136
137   JSplitPane jobPanel = new JSplitPane();
138
139
140   JScrollPane jobOptionsPane = new JScrollPane();
141
142   JPanel jobParameters = new JPanel();
143
144   JButton createpref = new JButton();
145
146   JButton deletepref = new JButton();
147
148   JButton revertpref = new JButton();
149
150   JButton updatepref = new JButton();
151
152   JButton startjob = new JButton();
153
154   JButton canceljob = new JButton();
155
156   JComboBox setName = new JComboBox();
157
158   JTextArea setDescr = new JTextArea();
159
160   JScrollPane paramPane = new JScrollPane();
161
162
163   JPanel optsAndparams = new JPanel();
164
165   RunnerConfig serviceOptions;
166
167   ParamDatastoreI paramStore;
168
169   private int MAX_OPTWIDTH = 200;
170
171   WsJobParameters(Jws2Instance service)
172   {
173     this(service, null);
174   }
175
176   public WsJobParameters(Jws2Instance service, WsParamSetI preset)
177   {
178     this(null, service, preset, null);
179   }
180
181   /**
182    * 
183    * @param desktop
184    *          - if null, create new JFrame outside of desktop
185    * @param service
186    * @param preset
187    */
188   public WsJobParameters(JFrame parent, Jws2Instance service,
189           WsParamSetI preset, List<Argument> jobArgset)
190   {
191     this(parent, null, service, preset, jobArgset);
192   }
193
194   /**
195    * 
196    * @param parent
197    * @param paramStorei
198    * @param service
199    * @param preset
200    * @param jobArgset
201    */
202   public WsJobParameters(JFrame parent, ParamDatastoreI paramStorei,
203           Jws2Instance service, WsParamSetI preset, List<Argument> jobArgset)
204   {
205     super();
206     jbInit();
207     this.paramStore = paramStorei;
208     if (paramStore == null)
209     {
210       paramStore = service.getParamStore();
211     }
212     this.service = service;
213     // argSetModified(false);
214     // populate parameter table
215     initForService(service, preset, jobArgset);
216     // display in new JFrame attached to parent.
217     validate();
218   }
219
220   int response = -1;
221
222   JDialog frame = null;
223
224   /**
225    * shows a modal dialog containing the parameters.
226    * 
227    * @return
228    */
229   public boolean showRunDialog()
230   {
231
232     frame = new JDialog(Desktop.instance, true);
233
234     frame.setTitle("Edit parameters for " + service.getActionText());
235     Rectangle deskr = Desktop.instance.getBounds();
236     Dimension pref = this.getPreferredSize();
237     frame.setBounds(new Rectangle((int) (deskr.getCenterX() - pref.width/2),
238             (int) (deskr.getCenterY() - pref.height/2), pref.width, pref.height));
239     frame.setContentPane(this);
240     
241     // should perhaps recover defaults from user prefs.
242
243     frame.validate();
244     javax.swing.SwingUtilities.invokeLater(new Runnable()
245     {
246         public void run()
247         {
248           jobPanel.setDividerLocation(0.25); 
249
250         }
251       });
252   frame.setVisible(true);
253
254     if (response > 0)
255     {
256       return true;
257     }
258     return false;
259   }
260
261   private void jbInit()
262   {
263     updatepref = JvSwingUtils.makeButton("Update",
264             "Update this existing user parameter set.",
265             new ActionListener()
266             {
267
268               public void actionPerformed(ActionEvent e)
269               {
270                 update_actionPerformed(e);
271               }
272             });
273     deletepref = JvSwingUtils.makeButton("Delete",
274             "Delete the currently selected user parameter set.",
275             new ActionListener()
276             {
277
278               public void actionPerformed(ActionEvent e)
279               {
280                 delete_actionPerformed(e);
281               }
282             });
283     createpref = JvSwingUtils.makeButton("Create",
284             "Create a new parameter set with the current settings.",
285             new ActionListener()
286             {
287
288               public void actionPerformed(ActionEvent e)
289               {
290                 create_actionPerformed(e);
291               }
292             });
293     revertpref = JvSwingUtils.makeButton("Revert",
294             "Undo all changes to the current parameter set",
295             new ActionListener()
296             {
297
298               public void actionPerformed(ActionEvent e)
299               {
300                 revert_actionPerformed(e);
301               }
302             });
303     startjob = JvSwingUtils.makeButton("Start Job",
304             "Start Job with current settings.", new ActionListener()
305             {
306               public void actionPerformed(ActionEvent e)
307               {
308                 startjob_actionPerformed(e);
309               }
310             });
311     canceljob = JvSwingUtils.makeButton("Cancel Job",
312             "Close this dialog and cancel job.", new ActionListener()
313             {
314               public void actionPerformed(ActionEvent e)
315               {
316                 canceljob_actionPerformed(e);
317               }
318             });
319
320     setDetails.setBorder(new TitledBorder("Details"));
321     setDetails.setLayout(new BorderLayout());
322     setDescr.setColumns(40);
323     setDescr.setWrapStyleWord(true);
324     setDescr.setLineWrap(true);
325     setDescr.setBackground(getBackground());
326     setDescr.setEditable(true);
327     setDescr.getDocument().addDocumentListener(this);
328     setDescr.setToolTipText("Click to edit the notes for this parameter set.");
329     JScrollPane setDescrView = new JScrollPane();
330     // setDescrView.setPreferredSize(new Dimension(350, 200));
331     setDescrView.getViewport().setView(setDescr);
332     setName.setEditable(true);
333     setName.addItemListener(this);
334     setName.getEditor().addActionListener(this);
335     JPanel setNameInfo = new JPanel(new FlowLayout(FlowLayout.LEFT));
336     GridBagLayout gbl = new GridBagLayout();
337     SetNamePanel.setLayout(gbl);
338
339     JLabel setNameLabel = new JLabel("Current parameter set name :");
340     setNameLabel.setFont(new java.awt.Font("Verdana", Font.PLAIN, 10));
341     
342     setNameInfo.add(setNameLabel);
343     setNameInfo.add(setName);
344     
345     // initial button visibility
346     updatepref.setVisible(false);
347     deletepref.setVisible(false);
348     revertpref.setVisible(false);
349     createpref.setVisible(false);
350     JPanel setsavebuts = new JPanel();
351     setsavebuts.setLayout(new FlowLayout(FlowLayout.LEFT)); // GridLayout(1,2));
352     ((FlowLayout) setsavebuts.getLayout()).setHgap(10);
353     ((FlowLayout) setsavebuts.getLayout()).setVgap(0);
354     JPanel spacer = new JPanel();
355     spacer.setPreferredSize(new Dimension(2,30));
356     setsavebuts.add(spacer);
357     setsavebuts.add(deletepref);
358     setsavebuts.add(revertpref);
359     setsavebuts.add(createpref);
360     setsavebuts.add(updatepref);
361 //    setsavebuts.setSize(new Dimension(150, 30));
362     JPanel buttonArea = new JPanel(new GridLayout(1, 1));
363     buttonArea.add(setsavebuts);
364     SetNamePanel.add(setNameInfo);
365     GridBagConstraints gbc = new GridBagConstraints();
366     gbc.gridheight = 2;
367     gbl.setConstraints(setNameInfo, gbc);
368     SetNamePanel.add(buttonArea);
369     gbc = new GridBagConstraints();
370     gbc.gridx = 0;
371     gbc.gridy = 2;
372     gbc.gridheight = 1;
373     gbl.setConstraints(buttonArea, gbc);
374     setDetails.add(setDescrView, BorderLayout.CENTER);
375
376     jobParameters.setBorder(new TitledBorder("Parameters"));
377     paramPane.setPreferredSize(new Dimension(360, 400));
378     jobOptions.setBorder(new TitledBorder("Options"));
379     
380     paramList.setBorder(new TitledBorder("Parameters"));
381     
382     JPanel bjo=new JPanel(new BorderLayout()),bjp=new JPanel(new BorderLayout());
383     bjo.add(jobOptions, BorderLayout.CENTER);
384     bjp.add(paramList, BorderLayout.CENTER);
385     optsAndparams.setLayout(new BorderLayout()); 
386
387     optsAndparams.add(bjo, BorderLayout.CENTER);
388     optsAndparams.add(bjp, BorderLayout.SOUTH);
389     paramPane.setViewportView(optsAndparams);
390  
391     setLayout(new BorderLayout());
392     jobPanel.setLeftComponent(setDetails);
393     jobPanel.setRightComponent(paramPane);
394     jobPanel.setOrientation(JSplitPane.VERTICAL_SPLIT);
395
396     add(SetNamePanel, BorderLayout.NORTH);
397     add(jobPanel, BorderLayout.CENTER);
398
399     JPanel dialogpanel = new JPanel();
400     dialogpanel.add(startjob);
401     dialogpanel.add(canceljob);
402     add(dialogpanel, BorderLayout.SOUTH);
403   }
404
405   protected void revert_actionPerformed(ActionEvent e)
406   {
407     reInitDialog(lastParmSet);
408
409   }
410
411   protected void update_actionPerformed(ActionEvent e)
412   {
413     if (isUserPreset)
414     {
415       String curname = ((String) setName.getSelectedItem()).trim();
416       _updatePreset(lastParmSet, curname);
417       lastParmSet = curname;
418       isUserPreset = true;
419       initArgSetModified();
420       syncSetNamesWithStore();
421     }
422   }
423
424   private void _deleteUserPreset(String lastParmSet2)
425   {
426     paramStore.deletePreset(lastParmSet2);
427   }
428
429   protected void delete_actionPerformed(ActionEvent e)
430   {
431     if (isUserPreset)
432     {
433       // delete current preset's saved entry
434       _deleteUserPreset(lastParmSet);
435     }
436     reInitDialog(null); // service default
437   }
438
439   protected void create_actionPerformed(ActionEvent e)
440   {
441     String curname = ((String) setName.getSelectedItem()).trim();
442     if (curname.length() > 0)
443     {
444       _storeCurrentPreset(curname);
445       lastParmSet = curname;
446       isUserPreset = true;
447       initArgSetModified();
448     }
449     else
450     {
451       // TODO: show warning
452       System.err.println("Invalid name. Not saved.");
453     }
454   }
455
456   protected void canceljob_actionPerformed(ActionEvent e)
457   {
458     response = 0;
459     if (frame != null)
460     {
461       frame.setVisible(false);
462     }
463   }
464
465   protected void startjob_actionPerformed(ActionEvent e)
466   {
467     response = 1;
468     if (frame != null)
469     {
470       frame.setVisible(false);
471     }
472   }
473
474   Jws2Instance service;
475
476   /**
477    * list of service presets in the gui
478    */
479   Hashtable servicePresets = null;
480
481   /**
482    * set if dialog is being set - so handlers will avoid spurious events
483    */
484   boolean settingDialog = false;
485
486   void initForService(Jws2Instance service, WsParamSetI jabap,
487           List<Argument> jabajobArgset)
488   {
489     WsParamSetI p = null;
490     List<ArgumentI> jobArgset = null;
491     settingDialog = true;
492     { // instantiate the abstract proxy for Jaba objects
493       jobArgset = jabajobArgset == null ? null : JabaParamStore
494               .getJwsArgsfromJaba(jabajobArgset);
495       p = jabap; // (jabap != null) ? paramStore.getPreset(jabap.getName()) :
496                  // null;
497     }
498
499     Hashtable exnames = new Hashtable();
500     for (int i = 0, iSize = setName.getItemCount(); i < iSize; i++)
501     {
502       exnames.put((String) setName.getItemAt(i), setName.getItemAt(i));
503     }
504     servicePresets = new Hashtable();
505     // Add the default entry - if not present already.
506     if (!exnames.contains(SVC_DEF))
507     {
508       setName.addItem(SVC_DEF);
509       exnames.put(SVC_DEF, SVC_DEF);
510       servicePresets.put(SVC_DEF, SVC_DEF);
511     }
512     String curname = (p == null ? "" : p.getName());
513     for (WsParamSetI pr : paramStore.getPresets())
514     {
515       if (!pr.isModifiable())
516       {
517         servicePresets.put(pr.getName(), "preset");
518       }
519       else
520       {
521       }
522       if (!exnames.contains(pr.getName()))
523       {
524         setName.addItem(pr.getName());
525       }
526     }
527     // TODO: if initial jobArgset matches a given user setting or preset then
528     // should recover setting accordingly
529     // updateTable(p, jobArgset);
530     if (p != null)
531     {
532       reInitDialog(p.getName());
533       initArgSetModified();
534     }
535     else
536     {
537       if (jobArgset != null && jobArgset.size() > 0)
538       {
539         curSetName = "Supplied Settings";
540         updateTable(p, jobArgset);
541       }
542       else
543       {
544         curSetName = null;
545         reInitDialog(null);
546       }
547     }
548     settingDialog = false;
549
550   }
551
552   @SuppressWarnings("unchecked")
553   private void updateTable(WsParamSetI p, List<ArgumentI> jobArgset)
554   {
555     // populate table from default parameter set.
556     List<ArgumentI> args = paramStore.getServiceParameters();
557
558     // split to params and required arguments
559     {
560       for (ArgumentI myarg : args)
561       {
562         // Ideally, Argument would implement isRequired !
563         if (myarg instanceof ParameterI)
564         {
565           ParameterI parm = (ParameterI) myarg;
566           paramList.add(opanp.addParameter(parm));
567         }
568         else
569         {
570           if (myarg instanceof OptionI)
571           {
572             OptionI opt = (OptionI) myarg;
573             OptionBox ob = opanp.addOption(opt);
574             jobOptions.add(ob, FlowLayout.LEFT);
575             ob.resetToDefault();
576             if (MAX_OPTWIDTH < ob.getPreferredSize().width)
577             {
578               MAX_OPTWIDTH = ob.getPreferredSize().width;
579             }
580
581           }
582           else
583           {
584             System.err.println("Ignoring unknown service argument type "
585                     + myarg.getClass().getName());
586           }
587         }
588       }
589       args = null; // no more args to process.
590     }
591     if (p != null)
592     {
593       isUserPreset = false;
594       // initialise setname
595       setName.setSelectedItem(lastSetName = p.getName());
596       setDescr.setText(lastDescrText = p.getDescription());
597       // TODO - URL link
598       try
599       {
600         args = p.getArguments();
601       } catch (Exception e)
602       {
603         e.printStackTrace();
604       }
605       // TODO: check if args should be unselected prior to resetting using the
606       // preset
607     }
608     else
609     {
610       if (lastParmSet == null)
611       {
612         isUserPreset = false;
613         // first call - so create a dummy name
614
615         setName.setSelectedItem(lastSetName = SVC_DEF);
616       }
617     }
618
619     if (jobArgset != null)
620     {
621       argSetModified(jobArgset, true);
622       args = jobArgset;
623     }
624     // get setargs from current object
625     if (args != null)
626     {
627       for (ArgumentI arg : args)
628       {
629         if (arg instanceof ParameterI)
630         {
631           opanp.setParameter((ParameterI) arg);
632         }
633         else
634         {
635           if (arg instanceof OptionI)
636           {
637             // System.out.println("Setting option "
638             // + System.identityHashCode(arg) + ":" + arg.getName()
639             // + " with " + arg.getDefaultValue());
640             opanp.selectOption((OptionI) arg, arg.getDefaultValue());
641           }
642         }
643
644       }
645     }
646
647     refreshParamLayout();
648     revalidate();
649   }
650
651   private boolean isModified()
652   {
653     return modifiedElements.size() > 0;
654   }
655
656   private Hashtable modifiedElements = new Hashtable();
657
658   /**
659    * reset gui and modification state settings
660    */
661   private void initArgSetModified()
662   {
663     curSetName = null;
664     modifiedElements.clear();
665     updateButtonDisplay();
666   }
667
668   private void updateButtonDisplay()
669   {
670     boolean _update = false, _create = false, _delete = false, _revert = false;
671     if (modifiedElements.size() > 0)
672     {
673       // set modified
674       _revert = true;
675       _update = isUserPreset; // can only update user presets
676       if (!isUserPreset || modifiedElements.containsKey(setName))
677       {
678         // name modified - can create new preset
679         _create = true;
680       }
681     }
682     else
683     {
684       // set unmodified
685     }
686     // can still delete a user preset
687     _delete = isUserPreset;
688
689     createpref.setVisible(_create);
690     updatepref.setVisible(_update);
691     deletepref.setVisible(_delete);
692     revertpref.setVisible(_revert);
693     validate();
694   }
695
696   public void argSetModified(Object modifiedElement, boolean b)
697   {
698     if (settingDialog)
699     {
700       return;
701     }
702     if (!b)
703     {
704       modifiedElements.remove(modifiedElement);
705     }
706     else
707     {
708       if (b && modifiedElement == setName
709               && modifiedElements.contains(modifiedElement))
710       {
711         // HACK! prevents iteration on makeSetNameValid
712         b = false;
713       }
714       modifiedElements.put(modifiedElement, modifiedElement);
715     }
716     // set mod status based on presence of elements in table
717     if (b && modifiedElements.size() > 0)
718     {
719       makeSetNameValid(!isUserPreset);
720       SetNamePanel.revalidate();
721     }
722     updateButtonDisplay();
723   }
724
725   private boolean isServicePreset(String selectedItem)
726   {
727     return selectedItem.equals(SVC_DEF)
728             || servicePresets.containsKey(selectedItem);
729   }
730
731   /**
732    * check if the current set name is a valid set name for saving, if not, then
733    * fix it.
734    */
735   private void makeSetNameValid(boolean newuserset)
736   {
737     boolean stn = settingDialog;
738     boolean renamed = false;
739     settingDialog = true;
740     String nm = (curSetName != null ? curSetName : (String) setName
741             .getSelectedItem());
742     // check if the name is reserved - if it is, rename it.
743     if (isServicePreset(nm))
744     {
745       nm = "User " + nm;
746       renamed = true;
747     }
748     String tnm = nm;
749     if (newuserset)
750     {
751       int i = 0;
752       while (paramStore.getPreset(tnm) != null)
753       {
754         tnm = nm + " (" + (++i) + ")";
755         renamed = true;
756       }
757       if (i > 0)
758       {
759         nm = tnm;
760       }
761     }
762
763     boolean makeupdate = false;
764     // sync the gui with the preset database
765     for (int i = 0, iS = setName.getItemCount(); i < iS; i++)
766     {
767       String snm = (String) setName.getItemAt(i);
768       if (snm.equals(nm))
769       {
770         makeupdate = true;
771         // setName.setSelectedIndex(i);
772       }
773     }
774     if (!makeupdate)
775     {
776       setName.addItem(curSetName = nm);
777       setName.setSelectedItem(curSetName);
778     }
779     if (renamed)
780     {
781       settingDialog = false; // we need this name change to be registered.
782       argSetModified(setName, renamed);
783     }
784     settingDialog = stn;
785   }
786
787
788   public void refreshParamLayout()
789   {
790     FlowLayout fl = new FlowLayout(FlowLayout.LEFT);
791     int sep=fl.getVgap();
792     int os=0,s = jobOptions.getBorder().getBorderInsets(jobOptions).bottom+jobOptions.getBorder().getBorderInsets(jobOptions).top+2 * sep;
793     int w = 2 * fl.getHgap() + (MAX_OPTWIDTH > OptsAndParamsPage.PARAM_WIDTH ? MAX_OPTWIDTH : OptsAndParamsPage.PARAM_WIDTH);
794     jobOptions.setLayout(fl);
795     if (opanp.getOptSet().size() > 0)
796     {
797       
798       for (OptionBox pbox : opanp.getOptSet().values())
799       {
800           pbox.revalidate();
801         s += sep + pbox.getPreferredSize().height; 
802       }
803       jobOptions.setPreferredSize(new Dimension(w, s));
804       jobOptions.setLayout(new GridLayout(opanp.getOptSet().size(),1));
805       os=s;
806     }
807     else
808     {
809       jobOptions.setVisible(false);
810     }
811
812     // Now layout the parameters assuming they occupy one column - to calculate total height of options+parameters
813     fl = new FlowLayout(FlowLayout.CENTER);
814     // helpful hint from http://stackoverflow.com/questions/2743177/top-alignment-for-flowlayout
815     fl.setAlignOnBaseline(true);
816     if (opanp.getParamSet().size() > 0)
817     {
818       paramList.setLayout(fl);
819
820       s = 2 * sep;
821       for (ParamBox pbox : opanp.getParamSet().values())
822       {
823         pbox.validate();
824         s += sep + pbox.getPreferredSize().height+pbox.getBorder().getBorderInsets(pbox).bottom; 
825       }
826       
827       paramList.setPreferredSize(new Dimension(w, s));
828       os+=s+2*sep+paramList.getBorder().getBorderInsets(paramList).bottom+paramList.getBorder().getBorderInsets(paramList).top;
829     }
830     else
831     {
832       paramList.setVisible(false);
833     }
834     // TODO: waste some time trying to eliminate any unnecessary .validate calls here
835     paramList.validate();
836     jobOptions.validate();
837     // System.out.println("Size will be : "+w+","+os);
838     optsAndparams.validate();
839     paramPane.getViewport().validate();
840     paramPane.getVerticalScrollBar().setBlockIncrement(OptsAndParamsPage.PARAM_CLOSEDHEIGHT*2);
841     paramPane.validate();
842     validate();
843   }
844
845   /**
846    * testing method - grab a service and parameter set and show the window
847    * 
848    * @param args
849    */
850   public static void main(String[] args)
851   {
852     jalview.ws.jws2.Jws2Discoverer disc = jalview.ws.jws2.Jws2Discoverer
853             .getDiscoverer();
854     int p = 0;
855     if (args.length > 0)
856     {
857       Vector<String> services = new Vector<String>();
858       services.addElement(args[p++]);
859       Jws2Discoverer.setServiceUrls(services);
860     }
861     try
862     {
863       disc.run();
864     } catch (Exception e)
865     {
866       System.err.println("Aborting. Problem discovering services.");
867       e.printStackTrace();
868       return;
869     }
870     Jws2Discoverer.Jws2Instance lastserv = null;
871     for (Jws2Discoverer.Jws2Instance service : disc.getServices())
872     {
873       lastserv = service;
874       if (p >= args.length || service.serviceType.equalsIgnoreCase(args[p]))
875       {
876         if (lastserv != null)
877         {
878           List<Preset> prl = null;
879           Preset pr = null;
880           if (++p < args.length)
881           {
882             PresetManager prman = lastserv.getPresets();
883             if (prman != null)
884             {
885               pr = prman.getPresetByName(args[p]);
886               if (pr == null)
887               {
888                 // just grab the last preset.
889                 prl = prman.getPresets();
890               }
891             }
892           }
893           else
894           {
895             PresetManager prman = lastserv.getPresets();
896             if (prman != null)
897             {
898               prl = prman.getPresets();
899             }
900           }
901           Iterator<Preset> en = (prl == null) ? null : prl.iterator();
902           while (en != null && en.hasNext())
903           {
904             if (en != null)
905             {
906               if (!en.hasNext())
907               {
908                 en = prl.iterator();
909               }
910               pr = en.next();
911             }
912             {
913               System.out.println("Testing opts dupes for "
914                       + lastserv.getUri() + " : "
915                       + lastserv.getActionText() + ":" + pr.getName());
916               List<Option> rg = lastserv.getRunnerConfig().getOptions();
917               for (Option o : rg)
918               {
919                 try
920                 {
921                   Option cpy = jalview.ws.jws2.ParameterUtils.copyOption(o);
922                 } catch (Exception e)
923                 {
924                   System.err.println("Failed to copy " + o.getName());
925                   e.printStackTrace();
926                 } catch (Error e)
927                 {
928                   System.err.println("Failed to copy " + o.getName());
929                   e.printStackTrace();
930                 }
931               }
932             }
933             {
934               System.out.println("Testing param dupes:");
935               List<Parameter> rg = lastserv.getRunnerConfig()
936                       .getParameters();
937               for (Parameter o : rg)
938               {
939                 try
940                 {
941                   Parameter cpy = jalview.ws.jws2.ParameterUtils
942                           .copyParameter(o);
943                 } catch (Exception e)
944                 {
945                   System.err.println("Failed to copy " + o.getName());
946                   e.printStackTrace();
947                 } catch (Error e)
948                 {
949                   System.err.println("Failed to copy " + o.getName());
950                   e.printStackTrace();
951                 }
952               }
953             }
954             {
955               System.out.println("Testing param write:");
956               List<String> writeparam = null, readparam = null;
957               try
958               {
959                 writeparam = jalview.ws.jws2.ParameterUtils
960                         .writeParameterSet(
961                                 pr.getArguments(lastserv.getRunnerConfig()),
962                                 " ");
963                 System.out.println("Testing param read :");
964                 List<Option> pset = jalview.ws.jws2.ParameterUtils
965                         .processParameters(writeparam,
966                                 lastserv.getRunnerConfig(), " ");
967                 readparam = jalview.ws.jws2.ParameterUtils
968                         .writeParameterSet(pset, " ");
969                 Iterator<String> o = pr.getOptions().iterator(), s = writeparam
970                         .iterator(), t = readparam.iterator();
971                 boolean failed = false;
972                 while (s.hasNext() && t.hasNext())
973                 {
974                   String on = o.next(), sn = s.next(), st = t.next();
975                   if (!sn.equals(st))
976                   {
977                     System.out.println("Original was " + on
978                             + " Phase 1 wrote " + sn + "\tPhase 2 wrote "
979                             + st);
980                     failed = true;
981                   }
982                 }
983                 if (failed)
984                 {
985                   System.out.println("Original parameters:\n"
986                           + pr.getOptions());
987                   System.out.println("Wrote parameters in first set:\n"
988                           + writeparam);
989                   System.out.println("Wrote parameters in second set:\n"
990                           + readparam);
991
992                 }
993               } catch (Exception e)
994               {
995                 e.printStackTrace();
996               }
997             }
998             WsJobParameters pgui = new WsJobParameters(lastserv,
999                     new JabaPreset(lastserv, pr));
1000             JFrame jf = new JFrame("Parameters for "
1001                     + lastserv.getActionText());
1002             JPanel cont = new JPanel(new BorderLayout());
1003             pgui.validate();
1004             cont.setPreferredSize(pgui.getPreferredSize());
1005             cont.add(pgui, BorderLayout.CENTER);
1006             jf.setLayout(new BorderLayout());
1007             jf.add(cont, BorderLayout.CENTER);
1008             jf.validate();
1009             final Thread thr = Thread.currentThread();
1010             jf.addWindowListener(new WindowListener()
1011             {
1012
1013               public void windowActivated(WindowEvent e)
1014               {
1015                 // TODO Auto-generated method stub
1016
1017               }
1018
1019               public void windowClosed(WindowEvent e)
1020               {
1021               }
1022
1023               public void windowClosing(WindowEvent e)
1024               {
1025                 thr.interrupt();
1026
1027               }
1028
1029               public void windowDeactivated(WindowEvent e)
1030               {
1031                 // TODO Auto-generated method stub
1032
1033               }
1034
1035               public void windowDeiconified(WindowEvent e)
1036               {
1037                 // TODO Auto-generated method stub
1038
1039               }
1040
1041               public void windowIconified(WindowEvent e)
1042               {
1043                 // TODO Auto-generated method stub
1044
1045               }
1046
1047               public void windowOpened(WindowEvent e)
1048               {
1049                 // TODO Auto-generated method stub
1050
1051               }
1052
1053             });
1054             jf.setVisible(true);
1055             boolean inter = false;
1056             while (!inter)
1057             {
1058               try
1059               {
1060                 Thread.sleep(10000);
1061               } catch (Exception e)
1062               {
1063                 inter = true;
1064               }
1065               ;
1066             }
1067             jf.dispose();
1068           }
1069         }
1070       }
1071     }
1072   }
1073
1074   public List<ArgumentI> getJobParams()
1075   {
1076     List<ArgumentI> argSet = new ArrayList<ArgumentI>();
1077     // recover options and parameters from GUI
1078     for (OptionBox opts : opanp.getOptSet().values())
1079     {
1080       OptionI opt = opts.getOptionIfEnabled();
1081       if (opt != null)
1082       {
1083         argSet.add(opt);
1084       }
1085     }
1086     for (ParamBox parambox : opanp.getParamSet().values())
1087     {
1088       ParameterI parm = parambox.getParameter();
1089       if (parm != null)
1090       {
1091         argSet.add(parm);
1092       }
1093     }
1094
1095     return argSet;
1096   }
1097
1098   String lastParmSet = null;
1099
1100   /*
1101    * Hashtable<String, Object[]> editedParams = new Hashtable<String,
1102    * Object[]>();
1103    * 
1104    * store the given parameters in the user parameter set database.
1105    * 
1106    * @param storeSetName - lastParmSet
1107    * 
1108    * @param descr - setDescr.getText()
1109    * 
1110    * @param jobParams - getJobParams()
1111    * 
1112    * private void _storeUserPreset(String storeSetName, String descr,
1113    * List<ArgumentI> jobParams) { // this is a simple hash store. Object[] pset;
1114    * editedParams.put(storeSetName, pset = new Object[3]); pset[0] =
1115    * storeSetName; pset[1] = descr; pset[2] = jobParams; // writeParam("Saving "
1116    * + storeSetName + ": ", jobParams); }
1117    * 
1118    * private void writeParam(String nm, List<ArgumentI> params) { for (ArgumentI
1119    * p : params) { System.out.println(nm + ":" + System.identityHashCode(p) +
1120    * " Name: " + p.getName() + " Value: " + p.getDefaultValue()); } }
1121    * 
1122    * private Object[] _getUserPreset(String setName) { Object[] pset =
1123    * editedParams.get(setName); // if (pset != null) // writeParam("Retrieving "
1124    * + setName + ": ", (List<Argument>) pset[2]); return pset; }
1125    * 
1126    * * remove the given user preset from the preset stash
1127    * 
1128    * @param setName
1129    * 
1130    * private void _deleteUserPreset(String setName) {
1131    * editedParams.remove(setName); }
1132    */
1133
1134   private void syncSetNamesWithStore()
1135   {
1136     int n = 0;
1137     // remove any set names in the drop down menu that aren't either a reserved
1138     // setting, or a user defined or service preset.
1139     Vector items = new Vector();
1140     while (n < setName.getItemCount())
1141     {
1142       String item = (String) setName.getItemAt(n);
1143       if (!item.equals(SVC_DEF) && !paramStore.presetExists(item))
1144       {
1145         setName.removeItemAt(n);
1146       }
1147       else
1148       {
1149         items.addElement(item);
1150         n++;
1151       }
1152     }
1153     if (!items.contains(SVC_DEF))
1154     {
1155       setName.addItem(SVC_DEF);
1156     }
1157     for (WsParamSetI upn : paramStore.getPresets())
1158     {
1159       if (!items.contains(upn.getName()))
1160       {
1161         setName.addItem(upn.getName());
1162       }
1163     }
1164   }
1165
1166   /**
1167    * true if lastParmSet is a user preset
1168    */
1169   boolean isUserPreset = false;
1170
1171   private void reInitDialog(String nextPreset)
1172   {
1173     settingDialog = true;
1174     // updateTable(null,null); // first reset to defaults
1175     WsParamSetI pset = null;
1176     if (nextPreset != null && nextPreset.length() > 0)
1177     {
1178       pset = paramStore.getPreset(nextPreset);
1179     }
1180     if (pset != null)
1181     {
1182       if (pset.isModifiable())
1183       {
1184         isUserPreset = true;
1185         setDescr.setText(pset.getDescription());
1186         updateTable(null, pset.getArguments());
1187         lastParmSet = nextPreset;
1188       }
1189       else
1190       {
1191         isUserPreset = false;
1192         setDescr.setText("");
1193         // must be a default preset from service
1194         updateTable(pset, null);
1195         lastParmSet = nextPreset;
1196       }
1197     }
1198     else
1199     {
1200       isUserPreset = false;
1201       // Service defaults
1202       setDescr.setText("");
1203       updateTable(null, null);
1204       lastParmSet = SVC_DEF;
1205     }
1206
1207     initArgSetModified();
1208     syncSetNamesWithStore();
1209     setName.setSelectedItem(lastParmSet);
1210     SetNamePanel.validate();
1211     validate();
1212     settingDialog = false;
1213
1214   }
1215
1216   String curSetName = null;
1217
1218   public void itemStateChanged(ItemEvent e)
1219   {
1220     if (e.getSource() == setName && e.getStateChange() == e.SELECTED)
1221     {
1222       final String setname = (String) setName.getSelectedItem();
1223       System.out.println("Item state changed for " + setname
1224               + " (handling ? " + !settingDialog + ")");
1225       if (settingDialog)
1226       {
1227         // ignore event
1228         return;
1229       }
1230       if (setname == null)
1231       {
1232         return;
1233       }
1234       javax.swing.SwingUtilities.invokeLater(new Runnable()
1235       {
1236         public void run()
1237         {
1238           doPreferenceComboStateChange(setname);
1239         }
1240       });
1241     }
1242   }
1243
1244   private void doPreferenceComboStateChange(String setname)
1245   {
1246     // user has selected a different item from combo-box
1247     if (isModified())
1248     {
1249       String lsetname = (curSetName != null) ? curSetName : lastParmSet;
1250       if (lsetname.equals(setname))
1251       {
1252         // setname was just edited - so ignore this event.
1253         return;
1254       }
1255       settingDialog = true;
1256       System.out.println("Prompting to save " + lsetname);
1257       if (javax.swing.JOptionPane
1258               .showConfirmDialog(
1259                       this,
1260                       "Parameter set '"
1261                               + lsetname
1262                               + "' is modifed, and your changes will be lost.\nReally change preset ?",
1263                       "Warning: Unsaved Changes",
1264                       javax.swing.JOptionPane.OK_CANCEL_OPTION) != JOptionPane.OK_OPTION)
1265       {
1266         // revert the combobox to the current item
1267         settingDialog = true;
1268         setName.setSelectedItem(lsetname);
1269         settingDialog = false;
1270         // and leave.
1271         return;
1272         // System.out.println("Saving for " + lsetname);
1273         // _storeCurrentPreset(lsetname);
1274
1275       }
1276     }
1277     settingDialog = true;
1278     reInitDialog(setname);
1279     settingDialog = false;
1280
1281   }
1282
1283   private void _renameExistingPreset(String oldName, String curSetName2)
1284   {
1285     paramStore.updatePreset(oldName, curSetName2, setDescr.getText(),
1286             getJobParams());
1287   }
1288
1289   /**
1290    * store current settings as given name. You should then reset gui.
1291    * 
1292    * @param curSetName2
1293    */
1294   private void _storeCurrentPreset(String curSetName2)
1295   {
1296     paramStore.storePreset(curSetName2, setDescr.getText(), getJobParams());
1297   }
1298
1299   private void _updatePreset(String lastParmSet2, String curname)
1300   {
1301     paramStore.updatePreset(lastParmSet2, curname, setDescr.getText(),
1302             getJobParams());
1303
1304   }
1305
1306   /**
1307    * last saved name for this user preset
1308    */
1309   String lastSetName = null;
1310
1311   /**
1312    * last saved value of the description text for this user preset
1313    */
1314   String lastDescrText = null;
1315
1316   public void actionPerformed(ActionEvent e)
1317   {
1318     if (e.getSource() instanceof Component)
1319     {
1320       Component src = (Component) e.getSource();
1321       if (src.getParent() == setName)
1322       {
1323         // rename any existing records we know about for this set.
1324         String newname = (String) e.getActionCommand().trim();
1325         String msg = null;
1326         if (isServicePreset(newname))
1327         {
1328           final String oldname = curSetName != null ? curSetName
1329                   : lastParmSet;
1330           final Component ourframe = this;
1331           settingDialog = true;
1332           setName.getEditor().setItem(oldname);
1333           settingDialog = false;
1334           javax.swing.SwingUtilities.invokeLater(new Runnable()
1335           {
1336             public void run()
1337             {
1338               JOptionPane.showMessageDialog(ourframe,
1339                       "Invalid name - preset already exists.",
1340                       "Invalid name", JOptionPane.WARNING_MESSAGE);
1341             }
1342           });
1343
1344           return;
1345         }
1346         curSetName = newname;
1347         System.err.println("New name for user setting " + curSetName
1348                 + " (was " + setName.getSelectedItem() + ")");
1349         if (curSetName.equals(setName.getSelectedItem()))
1350         {
1351           curSetName = null;
1352         }
1353         if (curSetName != null)
1354         {
1355           argSetModified(setName, true);
1356           return;
1357         }
1358
1359       }
1360     }
1361   }
1362
1363   private void checkDescrModified()
1364   {
1365     if (!settingDialog)
1366     {
1367
1368       argSetModified(
1369               setDescr,
1370               (lastDescrText == null ? setDescr.getText().trim().length() > 0
1371                       : !setDescr.getText().equals(lastDescrText)));
1372
1373     }
1374   }
1375
1376   public void insertUpdate(DocumentEvent e)
1377   {
1378     checkDescrModified();
1379   }
1380
1381   public void removeUpdate(DocumentEvent e)
1382   {
1383     checkDescrModified();
1384   }
1385
1386   public void changedUpdate(DocumentEvent e)
1387   {
1388     checkDescrModified();
1389   }
1390
1391   /**
1392    * 
1393    * @return null or the service preset selected by the user
1394    */
1395   public WsParamSetI getPreset()
1396   {
1397     if (isUserPreset || isModified()
1398             || (lastParmSet != null && lastParmSet.equals(SVC_DEF)))
1399     {
1400       return null;
1401     }
1402     else
1403     {
1404       return paramStore.getPreset(lastParmSet);
1405     }
1406   }
1407 }