2 * Jalview - A Sequence Alignment Editor and Viewer ($$Version-Rel$$)
3 * Copyright (C) $$Year-Rel$$ The Jalview Authors
5 * This file is part of Jalview.
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
10 * of the License, or (at your option) any later version.
12 * Jalview is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty
14 * of MERCHANTABILITY or FITNESS FOR A PARTICULAR
15 * PURPOSE. See the GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with Jalview. If not, see <http://www.gnu.org/licenses/>.
19 * The Jalview Authors are detailed in the 'AUTHORS' file.
24 import jalview.datamodel.HiddenColumns;
25 import jalview.io.cache.JvCacheableInputBox;
26 import jalview.schemes.AnnotationColourGradient;
27 import jalview.util.MessageManager;
28 import jalview.viewmodel.annotationfilter.AnnotationFilterParameter;
30 import java.awt.BorderLayout;
31 import java.awt.CardLayout;
32 import java.awt.Color;
33 import java.awt.Dimension;
34 import java.awt.event.ActionEvent;
35 import java.awt.event.ActionListener;
36 import java.awt.event.ItemEvent;
37 import java.awt.event.ItemListener;
38 import java.awt.event.KeyEvent;
39 import java.util.ArrayList;
41 import javax.swing.ButtonGroup;
42 import javax.swing.JCheckBox;
43 import javax.swing.JComboBox;
44 import javax.swing.JInternalFrame;
45 import javax.swing.JLayeredPane;
46 import javax.swing.JPanel;
47 import javax.swing.JRadioButton;
48 import javax.swing.border.TitledBorder;
50 import net.miginfocom.swing.MigLayout;
52 @SuppressWarnings("serial")
53 public class AnnotationColumnChooser extends AnnotationRowFilter implements
56 private JPanel switchableViewsPanel = new JPanel(new CardLayout());
58 private JPanel annotationComboBoxPanel = new JPanel();
60 private StructureFilterPanel gStructureFilterPanel;
62 private StructureFilterPanel ngStructureFilterPanel;
64 private StructureFilterPanel currentStructureFilterPanel;
66 private SearchPanel currentSearchPanel;
68 private SearchPanel gSearchPanel;
70 private SearchPanel ngSearchPanel;
72 private FurtherActionPanel currentFurtherActionPanel;
74 private FurtherActionPanel gFurtherActionPanel;
76 private FurtherActionPanel ngFurtherActionPanel;
78 public static final int ACTION_OPTION_SELECT = 1;
80 public static int ACTION_OPTION_HIDE = 2;
82 public static String NO_GRAPH_VIEW = "0";
84 public static String GRAPH_VIEW = "1";
86 private int actionOption = ACTION_OPTION_SELECT;
88 private HiddenColumns oldHiddenColumns;
90 protected int MIN_WIDTH = 420;
92 protected int MIN_HEIGHT = 430;
94 public AnnotationColumnChooser(AlignViewport av, final AlignmentPanel ap)
97 frame = new JInternalFrame();
98 frame.setContentPane(this);
99 frame.setLayer(JLayeredPane.PALETTE_LAYER);
100 Desktop.addInternalFrame(frame,
101 MessageManager.getString("label.select_by_annotation"), 520,
103 frame.setMinimumSize(new Dimension(MIN_WIDTH, MIN_HEIGHT));
105 addSliderChangeListener();
106 addSliderMouseListeners();
108 if (av.getAlignment().getAlignmentAnnotation() == null)
112 setOldHiddenColumns(av.getAlignment().getHiddenColumns());
115 setAnnotations(new JComboBox<>(getAnnotationItems(false)));
116 populateThresholdComboBox(threshold);
117 AnnotationColumnChooser lastChooser = av
118 .getAnnotationColumnSelectionState();
119 // restore Object state from the previous session if one exists
120 if (lastChooser != null)
122 currentSearchPanel = lastChooser
123 .getCurrentSearchPanel();
124 currentStructureFilterPanel = lastChooser
125 .getCurrentStructureFilterPanel();
126 annotations.setSelectedIndex(lastChooser
127 .getAnnotations().getSelectedIndex());
128 threshold.setSelectedIndex(lastChooser
129 .getThreshold().getSelectedIndex());
130 actionOption = lastChooser
132 percentThreshold.setSelected(lastChooser.percentThreshold
139 } catch (Exception ex)
150 protected void jbInit()
154 JPanel thresholdPanel = new JPanel();
155 thresholdPanel.setBorder(new TitledBorder(MessageManager
156 .getString("label.threshold_filter")));
157 thresholdPanel.setBackground(Color.white);
158 thresholdPanel.setFont(JvSwingUtils.getLabelFont());
159 thresholdPanel.setLayout(new MigLayout("", "[left][right]", "[][]"));
161 percentThreshold.setBackground(Color.white);
162 percentThreshold.setFont(JvSwingUtils.getLabelFont());
164 JPanel actionPanel = new JPanel();
165 actionPanel.setBackground(Color.white);
166 actionPanel.setFont(JvSwingUtils.getLabelFont());
168 JPanel graphFilterView = new JPanel();
169 graphFilterView.setLayout(new MigLayout("", "[left][right]", "[][]"));
170 graphFilterView.setBackground(Color.white);
172 JPanel noGraphFilterView = new JPanel();
173 noGraphFilterView.setLayout(new MigLayout("", "[left][right]", "[][]"));
174 noGraphFilterView.setBackground(Color.white);
176 annotationComboBoxPanel.setBackground(Color.white);
177 annotationComboBoxPanel.setFont(JvSwingUtils.getLabelFont());
179 gSearchPanel = new SearchPanel(this);
180 ngSearchPanel = new SearchPanel(this);
181 gFurtherActionPanel = new FurtherActionPanel(this);
182 ngFurtherActionPanel = new FurtherActionPanel(this);
183 gStructureFilterPanel = new StructureFilterPanel(this);
184 ngStructureFilterPanel = new StructureFilterPanel(this);
186 thresholdPanel.add(getThreshold());
187 thresholdPanel.add(percentThreshold, "wrap");
188 thresholdPanel.add(slider, "grow");
189 thresholdPanel.add(thresholdValue, "span, wrap");
192 actionPanel.add(cancel);
194 graphFilterView.add(gSearchPanel, "grow, span, wrap");
195 graphFilterView.add(gStructureFilterPanel, "grow, span, wrap");
196 graphFilterView.add(thresholdPanel, "grow, span, wrap");
197 graphFilterView.add(gFurtherActionPanel);
199 noGraphFilterView.add(ngSearchPanel, "grow, span, wrap");
200 noGraphFilterView.add(ngStructureFilterPanel, "grow, span, wrap");
201 noGraphFilterView.add(ngFurtherActionPanel);
203 annotationComboBoxPanel.add(getAnnotations());
204 switchableViewsPanel.add(noGraphFilterView,
205 AnnotationColumnChooser.NO_GRAPH_VIEW);
206 switchableViewsPanel.add(graphFilterView,
207 AnnotationColumnChooser.GRAPH_VIEW);
209 this.setLayout(new BorderLayout());
210 this.add(annotationComboBoxPanel, java.awt.BorderLayout.PAGE_START);
211 this.add(switchableViewsPanel, java.awt.BorderLayout.CENTER);
212 this.add(actionPanel, java.awt.BorderLayout.SOUTH);
214 selectedAnnotationChanged();
215 updateThresholdPanelToolTip();
219 protected void updateThresholdPanelToolTip()
221 thresholdValue.setToolTipText("");
222 slider.setToolTipText("");
224 String defaultTtip = MessageManager
225 .getString("info.change_threshold_mode_to_enable");
227 String thresh = getThreshold().getSelectedItem().toString();
228 if (thresh.equalsIgnoreCase("No Threshold"))
230 thresholdValue.setToolTipText(defaultTtip);
231 slider.setToolTipText(defaultTtip);
236 protected void reset()
238 if (this.getOldHiddenColumns() != null)
240 av.getColumnSelection().clear();
242 if (av.getAnnotationColumnSelectionState() != null)
244 HiddenColumns oldHidden = av
245 .getAnnotationColumnSelectionState()
246 .getOldHiddenColumns();
247 if (oldHidden != null)
249 ArrayList<int[]> regions = oldHidden.getHiddenColumnsCopy();
250 for (int[] positions : regions)
252 av.hideColumns(positions[0], positions[1]);
255 // TODO not clear why we need to hide all the columns (above) if we are
256 // going to copy the hidden columns over wholesale anyway
257 av.getAlignment().setHiddenColumns(oldHidden);
260 ap.paintAlignment(true);
265 public void valueChanged(boolean updateAllAnnotation)
267 if (slider.isEnabled())
269 getCurrentAnnotation().threshold.value = slider.getValue() / 1000f;
271 propagateSeqAssociatedThreshold(updateAllAnnotation,
272 getCurrentAnnotation());
273 ap.paintAlignment(false);
278 public void updateView()
280 // Check if combobox is still adjusting
286 AnnotationFilterParameter filterParams = new AnnotationFilterParameter();
288 setCurrentAnnotation(av.getAlignment().getAlignmentAnnotation()[annmap[getAnnotations()
289 .getSelectedIndex()]]);
291 int selectedThresholdItem = getSelectedThresholdItem(getThreshold()
292 .getSelectedIndex());
294 slider.setEnabled(true);
295 thresholdValue.setEnabled(true);
296 percentThreshold.setEnabled(true);
298 if (selectedThresholdItem == AnnotationColourGradient.NO_THRESHOLD)
300 slider.setEnabled(false);
301 thresholdValue.setEnabled(false);
302 thresholdValue.setText("");
303 percentThreshold.setEnabled(false);
304 // build filter params
306 else if (selectedThresholdItem != AnnotationColourGradient.NO_THRESHOLD)
308 if (getCurrentAnnotation().threshold == null)
310 getCurrentAnnotation()
312 new jalview.datamodel.GraphLine(
313 (getCurrentAnnotation().graphMax - getCurrentAnnotation().graphMin) / 2f,
314 "Threshold", Color.black));
318 float range = getCurrentAnnotation().graphMax * 1000
319 - getCurrentAnnotation().graphMin * 1000;
321 slider.setMinimum((int) (getCurrentAnnotation().graphMin * 1000));
322 slider.setMaximum((int) (getCurrentAnnotation().graphMax * 1000));
323 slider.setValue((int) (getCurrentAnnotation().threshold.value * 1000));
325 setThresholdValueText();
327 slider.setMajorTickSpacing((int) (range / 10f));
328 slider.setEnabled(true);
329 thresholdValue.setEnabled(true);
332 // build filter params
334 .setThresholdType(AnnotationFilterParameter.ThresholdType.NO_THRESHOLD);
335 if (getCurrentAnnotation().isQuantitative())
338 .setThresholdValue(getCurrentAnnotation().threshold.value);
340 if (selectedThresholdItem == AnnotationColourGradient.ABOVE_THRESHOLD)
343 .setThresholdType(AnnotationFilterParameter.ThresholdType.ABOVE_THRESHOLD);
345 else if (selectedThresholdItem == AnnotationColourGradient.BELOW_THRESHOLD)
348 .setThresholdType(AnnotationFilterParameter.ThresholdType.BELOW_THRESHOLD);
353 updateThresholdPanelToolTip();
354 if (currentStructureFilterPanel != null)
356 if (currentStructureFilterPanel.alphaHelix.isSelected())
358 filterParams.setFilterAlphaHelix(true);
360 if (currentStructureFilterPanel.betaStrand.isSelected())
362 filterParams.setFilterBetaSheet(true);
364 if (currentStructureFilterPanel.turn.isSelected())
366 filterParams.setFilterTurn(true);
370 if (currentSearchPanel != null)
372 if (!currentSearchPanel.searchBox.getUserInput().isEmpty())
374 filterParams.setRegexString(currentSearchPanel.searchBox
376 if (currentSearchPanel.displayName.isSelected())
379 .addRegexSearchField(AnnotationFilterParameter.SearchableAnnotationField.DISPLAY_STRING);
381 if (currentSearchPanel.description.isSelected())
384 .addRegexSearchField(AnnotationFilterParameter.SearchableAnnotationField.DESCRIPTION);
389 // show hidden columns here, before changing the column selection in
390 // filterAnnotations, because showing hidden columns has the side effect of
391 // adding them to the selection
392 av.showAllHiddenColumns();
393 av.getColumnSelection().filterAnnotations(
394 getCurrentAnnotation().annotations, filterParams);
396 if (getActionOption() == ACTION_OPTION_HIDE)
398 av.hideSelectedColumns();
403 av.setAnnotationColumnSelectionState(this);
404 ap.paintAlignment(true);
407 public HiddenColumns getOldHiddenColumns()
409 return oldHiddenColumns;
412 public void setOldHiddenColumns(HiddenColumns currentHiddenColumns)
414 if (currentHiddenColumns != null)
416 this.oldHiddenColumns = new HiddenColumns(currentHiddenColumns);
420 public FurtherActionPanel getCurrentFutherActionPanel()
422 return currentFurtherActionPanel;
425 public void setCurrentFutherActionPanel(
426 FurtherActionPanel currentFutherActionPanel)
428 this.currentFurtherActionPanel = currentFutherActionPanel;
431 public SearchPanel getCurrentSearchPanel()
433 return currentSearchPanel;
436 public void setCurrentSearchPanel(SearchPanel currentSearchPanel)
438 this.currentSearchPanel = currentSearchPanel;
441 public int getActionOption()
446 public void setActionOption(int actionOption)
448 this.actionOption = actionOption;
451 public StructureFilterPanel getCurrentStructureFilterPanel()
453 return currentStructureFilterPanel;
456 public void setCurrentStructureFilterPanel(
457 StructureFilterPanel currentStructureFilterPanel)
459 this.currentStructureFilterPanel = currentStructureFilterPanel;
462 public void select_action(ActionEvent actionEvent)
464 JRadioButton radioButton = (JRadioButton) actionEvent.getSource();
465 if (radioButton.isSelected())
467 setActionOption(ACTION_OPTION_SELECT);
472 public void hide_action(ActionEvent actionEvent)
474 JRadioButton radioButton = (JRadioButton) actionEvent.getSource();
475 if (radioButton.isSelected())
477 setActionOption(ACTION_OPTION_HIDE);
483 public void itemStateChanged(ItemEvent e)
485 selectedAnnotationChanged();
489 public void selectedAnnotationChanged()
491 String currentView = AnnotationColumnChooser.NO_GRAPH_VIEW;
492 if (av.getAlignment()
493 .getAlignmentAnnotation()[annmap[getAnnotations()
494 .getSelectedIndex()]].isQuantitative())
496 currentView = AnnotationColumnChooser.GRAPH_VIEW;
499 gSearchPanel.syncState();
500 gFurtherActionPanel.syncState();
501 gStructureFilterPanel.syncState();
503 ngSearchPanel.syncState();
504 ngFurtherActionPanel.syncState();
505 ngStructureFilterPanel.syncState();
507 CardLayout switchableViewsLayout = (CardLayout) switchableViewsPanel
509 switchableViewsLayout.show(switchableViewsPanel, currentView);
513 public class FurtherActionPanel extends JPanel
515 private AnnotationColumnChooser aColChooser;
517 private JRadioButton hideOption = new JRadioButton();
519 private JRadioButton selectOption = new JRadioButton();
521 private ButtonGroup optionsGroup = new ButtonGroup();
523 public FurtherActionPanel(AnnotationColumnChooser aColChooser)
525 this.aColChooser = aColChooser;
526 JvSwingUtils.jvInitComponent(selectOption, "action.select");
527 selectOption.addActionListener(new ActionListener()
530 public void actionPerformed(ActionEvent actionEvent)
532 selectRadioAction(actionEvent);
536 JvSwingUtils.jvInitComponent(hideOption, "action.hide");
537 hideOption.addActionListener(new ActionListener()
540 public void actionPerformed(ActionEvent actionEvent)
542 hideRadioAction(actionEvent);
546 optionsGroup.add(selectOption);
547 optionsGroup.add(hideOption);
548 optionsGroup.setSelected(selectOption.getModel(), true);
550 JvSwingUtils.jvInitComponent(this);
553 this.add(selectOption);
554 this.add(hideOption);
557 public void selectRadioAction(ActionEvent actionEvent)
559 aColChooser.setCurrentFutherActionPanel(this);
560 aColChooser.select_action(actionEvent);
563 public void hideRadioAction(ActionEvent actionEvent)
565 aColChooser.setCurrentFutherActionPanel(this);
566 aColChooser.hide_action(actionEvent);
569 public void syncState()
571 if (aColChooser.getActionOption() == AnnotationColumnChooser.ACTION_OPTION_HIDE)
573 this.optionsGroup.setSelected(this.hideOption.getModel(), true);
577 this.optionsGroup.setSelected(this.selectOption.getModel(), true);
582 public class StructureFilterPanel extends JPanel
584 private AnnotationColumnChooser aColChooser;
586 private JCheckBox alphaHelix = new JCheckBox();
588 private JCheckBox betaStrand = new JCheckBox();
590 private JCheckBox turn = new JCheckBox();
592 private JCheckBox all = new JCheckBox();
594 public StructureFilterPanel(AnnotationColumnChooser aColChooser)
596 this.aColChooser = aColChooser;
598 JvSwingUtils.jvInitComponent(alphaHelix, "label.alpha_helix");
599 alphaHelix.addActionListener(new ActionListener()
602 public void actionPerformed(ActionEvent actionEvent)
604 alphaHelix_actionPerformed();
608 JvSwingUtils.jvInitComponent(betaStrand, "label.beta_strand");
609 betaStrand.addActionListener(new ActionListener()
612 public void actionPerformed(ActionEvent actionEvent)
614 betaStrand_actionPerformed();
618 JvSwingUtils.jvInitComponent(turn, "label.turn");
619 turn.addActionListener(new ActionListener()
622 public void actionPerformed(ActionEvent actionEvent)
624 turn_actionPerformed();
628 JvSwingUtils.jvInitComponent(all, "label.select_all");
629 all.addActionListener(new ActionListener()
632 public void actionPerformed(ActionEvent actionEvent)
634 all_actionPerformed();
638 this.setBorder(new TitledBorder(MessageManager
639 .getString("label.structures_filter")));
640 JvSwingUtils.jvInitComponent(this);
643 this.add(alphaHelix);
644 this.add(betaStrand);
648 public void alphaHelix_actionPerformed()
650 updateSelectAllState();
651 aColChooser.setCurrentStructureFilterPanel(this);
652 aColChooser.updateView();
655 public void betaStrand_actionPerformed()
657 updateSelectAllState();
658 aColChooser.setCurrentStructureFilterPanel(this);
659 aColChooser.updateView();
662 public void turn_actionPerformed()
664 updateSelectAllState();
665 aColChooser.setCurrentStructureFilterPanel(this);
666 aColChooser.updateView();
669 public void all_actionPerformed()
671 if (all.isSelected())
673 alphaHelix.setSelected(true);
674 betaStrand.setSelected(true);
675 turn.setSelected(true);
679 alphaHelix.setSelected(false);
680 betaStrand.setSelected(false);
681 turn.setSelected(false);
683 aColChooser.setCurrentStructureFilterPanel(this);
684 aColChooser.updateView();
687 public void updateSelectAllState()
689 if (alphaHelix.isSelected() && betaStrand.isSelected()
690 && turn.isSelected())
692 all.setSelected(true);
696 all.setSelected(false);
700 public void syncState()
702 StructureFilterPanel sfp = aColChooser
703 .getCurrentStructureFilterPanel();
706 alphaHelix.setSelected(sfp.alphaHelix.isSelected());
707 betaStrand.setSelected(sfp.betaStrand.isSelected());
708 turn.setSelected(sfp.turn.isSelected());
709 if (sfp.all.isSelected())
711 all.setSelected(true);
712 alphaHelix.setSelected(true);
713 betaStrand.setSelected(true);
714 turn.setSelected(true);
721 public class SearchPanel extends JPanel
723 private AnnotationColumnChooser aColChooser;
725 private JCheckBox displayName = new JCheckBox();
727 private JCheckBox description = new JCheckBox();
729 private static final String FILTER_BY_ANN_CACHE_KEY = "CACHE.SELECT_FILTER_BY_ANNOT";
731 public JvCacheableInputBox<String> searchBox = new JvCacheableInputBox<>(
732 FILTER_BY_ANN_CACHE_KEY);
734 public SearchPanel(AnnotationColumnChooser aColChooser)
737 this.aColChooser = aColChooser;
738 JvSwingUtils.jvInitComponent(this);
739 this.setBorder(new TitledBorder(MessageManager
740 .getString("label.search_filter")));
742 searchBox.setPrototypeDisplayValue("XXXXXXXXXXXXXXXXXXXXXXX");
743 searchBox.setToolTipText(MessageManager
744 .getString("info.enter_search_text_here"));
745 searchBox.getEditor().getEditorComponent()
746 .addKeyListener(new java.awt.event.KeyAdapter()
749 public void keyPressed(KeyEvent e)
751 if (e.getKeyCode() == KeyEvent.VK_ENTER)
754 searchStringAction();
761 JvSwingUtils.jvInitComponent(displayName, "label.label");
762 displayName.addActionListener(new ActionListener()
765 public void actionPerformed(ActionEvent actionEvent)
767 displayNameCheckboxAction();
771 JvSwingUtils.jvInitComponent(description, "label.description");
772 description.addActionListener(new ActionListener()
775 public void actionPerformed(ActionEvent actionEvent)
777 discriptionCheckboxAction();
783 this.add(displayName);
784 this.add(description);
787 public void displayNameCheckboxAction()
789 aColChooser.setCurrentSearchPanel(this);
790 aColChooser.updateView();
793 public void discriptionCheckboxAction()
795 aColChooser.setCurrentSearchPanel(this);
796 aColChooser.updateView();
799 public void searchStringAction()
801 aColChooser.setCurrentSearchPanel(this);
802 aColChooser.updateView();
803 updateSearchPanelToolTips();
804 searchBox.updateCache();
807 public void syncState()
809 SearchPanel sp = aColChooser.getCurrentSearchPanel();
812 description.setEnabled(sp.description.isEnabled());
813 description.setSelected(sp.description.isSelected());
815 displayName.setEnabled(sp.displayName.isEnabled());
816 displayName.setSelected(sp.displayName.isSelected());
818 searchBox.setSelectedItem(sp.searchBox.getUserInput());
820 updateSearchPanelToolTips();
823 public void updateSearchPanelToolTips()
825 String defaultTtip = MessageManager
826 .getString("info.enter_search_text_to_enable");
827 String labelTtip = MessageManager.formatMessage(
828 "info.search_in_annotation_label", annotations
829 .getSelectedItem().toString());
830 String descTtip = MessageManager.formatMessage(
831 "info.search_in_annotation_description", annotations
832 .getSelectedItem().toString());
833 displayName.setToolTipText(displayName.isEnabled() ? labelTtip
835 description.setToolTipText(description.isEnabled() ? descTtip
841 public void ok_actionPerformed()
844 super.ok_actionPerformed();
848 public void cancel_actionPerformed()
851 super.cancel_actionPerformed();
854 private void saveCache()
856 gSearchPanel.searchBox.persistCache();
857 ngSearchPanel.searchBox.persistCache();
858 gSearchPanel.searchBox.updateCache();
859 ngSearchPanel.searchBox.updateCache();