3 * Copyright (C) 2014 EDIT
4 * European Distributed Institute of Taxonomy
5 * http://www.e-taxonomy.eu
7 * The contents of this file are subject to the Mozilla Public License Version 1.1
8 * See LICENSE.TXT at the top of this package for the full license terms.
10 package eu
.etaxonomy
.taxeditor
.editor
.molecular
;
13 import info
.bioinfweb
.libralign
.alignmentarea
.AlignmentArea
;
14 import info
.bioinfweb
.libralign
.alignmentarea
.selection
.SelectionModel
;
15 import info
.bioinfweb
.libralign
.dataarea
.implementations
.ConsensusSequenceArea
;
16 import info
.bioinfweb
.libralign
.dataarea
.implementations
.SequenceIndexArea
;
17 import info
.bioinfweb
.libralign
.dataarea
.implementations
.pherogram
.PherogramAlignmentModel
;
18 import info
.bioinfweb
.libralign
.dataarea
.implementations
.pherogram
.PherogramArea
;
19 import info
.bioinfweb
.libralign
.dataarea
.implementations
.pherogram
.ShiftChange
;
20 import info
.bioinfweb
.libralign
.editsettings
.EditSettingsChangeEvent
;
21 import info
.bioinfweb
.libralign
.editsettings
.EditSettingsListener
;
22 import info
.bioinfweb
.libralign
.multiplealignments
.AlignmentAreaList
;
23 import info
.bioinfweb
.libralign
.multiplealignments
.MultipleAlignmentsContainer
;
24 import info
.bioinfweb
.libralign
.pherogram
.provider
.BioJavaPherogramProvider
;
25 import info
.bioinfweb
.libralign
.pherogram
.provider
.PherogramProvider
;
26 import info
.bioinfweb
.libralign
.pherogram
.provider
.ReverseComplementPherogramProvider
;
27 import info
.bioinfweb
.libralign
.sequenceprovider
.SequenceDataChangeListener
;
28 import info
.bioinfweb
.libralign
.sequenceprovider
.SequenceDataProvider
;
29 import info
.bioinfweb
.libralign
.sequenceprovider
.SequenceUtils
;
30 import info
.bioinfweb
.libralign
.sequenceprovider
.adapters
.StringAdapter
;
31 import info
.bioinfweb
.libralign
.sequenceprovider
.events
.SequenceChangeEvent
;
32 import info
.bioinfweb
.libralign
.sequenceprovider
.events
.SequenceRenamedEvent
;
33 import info
.bioinfweb
.libralign
.sequenceprovider
.events
.TokenChangeEvent
;
34 import info
.bioinfweb
.libralign
.sequenceprovider
.implementations
.PackedSequenceDataProvider
;
35 import info
.bioinfweb
.libralign
.sequenceprovider
.tokenset
.BioJavaTokenSet
;
36 import info
.bioinfweb
.libralign
.sequenceprovider
.tokenset
.TokenSet
;
39 import java
.io
.IOException
;
40 import java
.io
.InputStream
;
42 import java
.util
.ArrayList
;
43 import java
.util
.Collection
;
44 import java
.util
.Collections
;
45 import java
.util
.Iterator
;
47 import java
.util
.TreeMap
;
49 import org
.biojava
.bio
.chromatogram
.ChromatogramFactory
;
50 import org
.biojava
.bio
.chromatogram
.UnsupportedChromatogramFormatException
;
51 import org
.biojava3
.core
.sequence
.compound
.DNACompoundSet
;
52 import org
.biojava3
.core
.sequence
.compound
.NucleotideCompound
;
53 import org
.eclipse
.core
.runtime
.IProgressMonitor
;
54 import org
.eclipse
.swt
.SWT
;
55 import org
.eclipse
.swt
.widgets
.Composite
;
56 import org
.eclipse
.ui
.IActionBars
;
57 import org
.eclipse
.ui
.IEditorInput
;
58 import org
.eclipse
.ui
.IEditorPart
;
59 import org
.eclipse
.ui
.IEditorSite
;
60 import org
.eclipse
.ui
.PartInitException
;
61 import org
.eclipse
.ui
.PlatformUI
;
62 import org
.eclipse
.ui
.commands
.ICommandService
;
63 import org
.eclipse
.ui
.part
.EditorPart
;
65 import eu
.etaxonomy
.cdm
.api
.conversation
.ConversationHolder
;
66 import eu
.etaxonomy
.cdm
.api
.service
.molecular
.ISequenceService
;
67 import eu
.etaxonomy
.cdm
.model
.media
.MediaUtils
;
68 import eu
.etaxonomy
.cdm
.model
.molecular
.Sequence
;
69 import eu
.etaxonomy
.cdm
.model
.molecular
.SequenceString
;
70 import eu
.etaxonomy
.cdm
.model
.molecular
.SingleRead
;
71 import eu
.etaxonomy
.cdm
.model
.molecular
.SingleReadAlignment
;
72 import eu
.etaxonomy
.taxeditor
.editor
.handler
.alignmenteditor
.ToggleInsertOverwriteHandler
;
73 import eu
.etaxonomy
.taxeditor
.editor
.handler
.alignmenteditor
.ToggleLeftRightInsertionHandler
;
74 import eu
.etaxonomy
.taxeditor
.store
.CdmStore
;
79 * Editor component to edit a contig alignment used to combine different overlapping pherograms from Sanger sequencing to
80 * a consensus sequence.
82 * The contained GUI components used to edit the alignment come from <a href="http://bioinfweb.info/LibrAlign/">LibrAlign</a>.
88 public class AlignmentEditor
extends EditorPart
{
89 public static final String ID
= "eu.etaxonomy.taxeditor.editor.molecular.AlignmentEditor";
91 public static final int READS_AREA_INDEX
= 1;
92 public static final int CONSENSUS_AREA_INDEX
= READS_AREA_INDEX
+ 1;
93 public static final int PHEROGRAM_AREA_INDEX
= 0;
94 public static final String DEFAULT_READ_NAME_PREFIX
= "Read ";
95 public static final String CONSENSUS_NAME
= "Consensus";
98 private final ConversationHolder conversationHolder
;
100 private final SequenceDataChangeListener DIRTY_LISTENER
= new SequenceDataChangeListener() {
102 public <T
> void afterTokenChange(TokenChangeEvent
<T
> e
) {
107 public <T
> void afterSequenceRenamed(SequenceRenamedEvent
<T
> e
) {
112 public <T
> void afterSequenceChange(SequenceChangeEvent
<T
> e
) {
117 public <T
, U
> void afterProviderChanged(SequenceDataProvider
<T
> oldProvider
,
118 SequenceDataProvider
<U
> newProvider
) { // Not expected.
124 private MultipleAlignmentsContainer alignmentsContainer
= null;
125 private final Map
<Integer
, SingleReadAlignment
> cdmMap
= new TreeMap
<Integer
, SingleReadAlignment
>(); //TODO Move this to ContigSequenceDataProvider
126 private boolean dirty
= false;
129 public AlignmentEditor() {
131 conversationHolder
= CdmStore
.createConversation();
135 private void refreshToolbarElement(String id
) {
136 ICommandService commandService
=
137 (ICommandService
)PlatformUI
.getWorkbench().getActiveWorkbenchWindow().getService(ICommandService
.class);
138 if (commandService
!= null) {
139 commandService
.refreshElements(id
, Collections
.EMPTY_MAP
);
144 private void registerEditSettingListener(MultipleAlignmentsContainer container
) {
145 container
.getEditSettings().addListener(new EditSettingsListener() {
147 public void workingModeChanged(EditSettingsChangeEvent e
) {} // Currently nothing to do
150 public void insertLeftInDataAreaChanged(EditSettingsChangeEvent e
) {
152 refreshToolbarElement(ToggleLeftRightInsertionHandler
.COMMAND_ID
);
156 public void insertChanged(EditSettingsChangeEvent e
) {
158 refreshToolbarElement(ToggleInsertOverwriteHandler
.COMMAND_ID
);
164 private AlignmentArea
createIndexArea(MultipleAlignmentsContainer container
) {
165 AlignmentArea result
= new AlignmentArea(container
);
166 result
.setAllowVerticalScrolling(false);
167 result
.getDataAreas().getTopAreas().add(new SequenceIndexArea(result
.getContentArea()));
172 private AlignmentArea
createEditableAlignmentArea(MultipleAlignmentsContainer container
, boolean allowVerticalScrolling
) {
173 AlignmentArea result
= new AlignmentArea(container
);
174 result
.setAllowVerticalScrolling(allowVerticalScrolling
);
176 TokenSet
<NucleotideCompound
> tokenSet
= new BioJavaTokenSet
<NucleotideCompound
>(new DNACompoundSet(), true);
177 SequenceDataProvider
<NucleotideCompound
> provider
= new PackedSequenceDataProvider
<NucleotideCompound
>(tokenSet
);
178 result
.setSequenceProvider(provider
, false);
179 provider
.getChangeListeners().add(DIRTY_LISTENER
);
185 private AlignmentArea
createConsensusHintArea(MultipleAlignmentsContainer container
,
186 SequenceDataProvider
<?
> sequenceProvider
) {
188 AlignmentArea result
= new AlignmentArea(container
);
189 result
.setAllowVerticalScrolling(false);
190 result
.getDataAreas().getBottomAreas().add(
191 new ConsensusSequenceArea(result
.getContentArea(), sequenceProvider
));
196 private MultipleAlignmentsContainer
getAlignmentsContainer() {
197 if (alignmentsContainer
== null) {
198 alignmentsContainer
= new MultipleAlignmentsContainer();
200 AlignmentAreaList list
= alignmentsContainer
.getAlignmentAreas();
201 list
.add(createIndexArea(alignmentsContainer
));
202 AlignmentArea readsArea
= createEditableAlignmentArea(alignmentsContainer
, true);
203 list
.add(readsArea
); // Make sure READS_AREA_INDEX is correct.
204 list
.add(createEditableAlignmentArea(alignmentsContainer
, false)); // Make sure COMSENSUS_AREA_INDEX is correct.
205 list
.add(createConsensusHintArea(alignmentsContainer
,
206 readsArea
.getSequenceProvider()));
208 registerEditSettingListener(alignmentsContainer
);
210 return alignmentsContainer
;
214 private AlignmentArea
getReadsArea() {
215 return getAlignmentsContainer().getAlignmentAreas().get(READS_AREA_INDEX
);
219 private AlignmentArea
getConsensusArea() {
220 return getAlignmentsContainer().getAlignmentAreas().get(CONSENSUS_AREA_INDEX
);
224 private PherogramArea
getPherogramArea(int sequenceID
) {
225 return (PherogramArea
)getReadsArea().getDataAreas().getSequenceAreas(sequenceID
).get(PHEROGRAM_AREA_INDEX
);
229 private void createTestContents() {
232 addRead(new File("D:/Users/BenStoever/Documents/Studium/Projekte/Promotion/EDITor/Quelltexte/LibrAlign branch/Repository/eu.etaxonomy.taxeditor.editor/src/main/resources/AlignmentTestData/JR430_JR-P01.ab1").toURI(), false);
233 addRead(new File("D:/Users/BenStoever/Documents/Studium/Projekte/Promotion/EDITor/Quelltexte/LibrAlign branch/Repository/eu.etaxonomy.taxeditor.editor/src/main/resources/AlignmentTestData/JR444_JR-P05.ab1").toURI(), false);
235 // Add test consensus sequence:
236 SequenceDataProvider consensusProvider
= getConsensusArea().getSequenceProvider();
237 int id
= consensusProvider
.addSequence(CONSENSUS_NAME
);
238 Collection
<Object
> tokens
= new ArrayList
<Object
>(); // First save tokens in a collection to avoid GUI updated for each token.
239 tokens
.add(consensusProvider
.getTokenSet().tokenByKeyChar('A'));
240 tokens
.add(consensusProvider
.getTokenSet().tokenByKeyChar('C'));
241 tokens
.add(consensusProvider
.getTokenSet().tokenByKeyChar('G'));
242 tokens
.add(consensusProvider
.getTokenSet().tokenByKeyChar('T'));
243 consensusProvider
.insertTokensAt(id
, 0, tokens
);
245 catch (Exception e
) {
246 throw new RuntimeException(e
);
251 private void readCDMData(Sequence sequenceNode
) {
252 //TODO If called from somewhere else than createPartControl() the editorInput needs to be checked and previous contents need to be cleared (or updated).
255 for (SingleReadAlignment singleReadAlignment
: sequenceNode
.getSingleReadAlignments()) {
257 SingleRead pherogramInfo
= singleReadAlignment
.getSingleRead();
258 int id
= addRead(pherogramInfo
.getPrimer().getLabel(), //TODO Should the sequence name contain other/additional/alternative data? Can the same string as in the derivative tree be used here?
259 MediaUtils
.getFirstMediaRepresentationPart(pherogramInfo
.getPherogram()).getUri(),
260 singleReadAlignment
.isReverseComplement(),
261 singleReadAlignment
.getEditedSequence(),
262 singleReadAlignment
.getShifts());
263 cdmMap
.put(id
, singleReadAlignment
);
265 catch (IOException e
) {
266 e
.printStackTrace(); //TODO Output to user (Possibly collect for all pherograms and display in the end.)
268 catch (UnsupportedChromatogramFormatException e
) {
269 e
.printStackTrace(); //TODO Output to user (Possibly collect for all pherograms and display in the end.)
273 // Set consensus sequence:
274 SequenceDataProvider consensusProvider
= getConsensusArea().getSequenceProvider();
275 int id
= consensusProvider
.addSequence(CONSENSUS_NAME
);
276 consensusProvider
.insertTokensAt(id
, 0, SequenceUtils
.stringToTokenList(
277 sequenceNode
.getConsensusSequence().getString(), consensusProvider
.getTokenSet()));
278 //TODO Can the consensus sequence also be null? / Should it be created here, if nothing is in the DB?
283 * @see org.eclipse.ui.part.WorkbenchPart#createPartControl(org.eclipse.swt.widgets.Composite)
286 public void createPartControl(Composite parent
) {
287 getAlignmentsContainer().createSWTWidget(parent
, SWT
.NONE
);
290 if (getEditorInput() instanceof AlignmentEditorInput
) {
291 if (((AlignmentEditorInput
)getEditorInput()).getSequenceNode() != null) {
292 Sequence sequenceNode
= ((AlignmentEditorInput
)getEditorInput()).getSequenceNode();
293 //re-load into the current session if it is already persisted in the DB
294 if(sequenceNode
!=null && sequenceNode
.getId()!=0){
295 sequenceNode
= CdmStore
.getService(ISequenceService
.class).load(sequenceNode
.getUuid());
297 readCDMData(sequenceNode
);
300 createTestContents(); // This case will removed after the test phase and an exception should probably be thrown.
304 throw new IllegalArgumentException("The editor input must have the type " +
305 AlignmentEditorInput
.class.getCanonicalName()); //TODO What should be done here?
310 private void updateStatusBar() {
311 IActionBars bars
= getEditorSite().getActionBars();
312 bars
.getStatusLineManager().setMessage("Edit mode: " +
313 (getReadsArea().getEditSettings().isInsert() ?
"Insert" : "Overwrite") + " " +
314 "Insertion in pherogram: " +
315 (getReadsArea().getEditSettings().isInsertLeftInDataArea() ?
"Left" : "Right"));
319 private SingleReadAlignment
.Shift
[] convertToCDMShifts(PherogramAlignmentModel alignmentModel
) {
320 SingleReadAlignment
.Shift
[] result
= new SingleReadAlignment
.Shift
[alignmentModel
.getShiftChangeCount()];
321 Iterator
<ShiftChange
> iterator
= alignmentModel
.shiftChangeIterator();
323 while (iterator
.hasNext()) {
324 ShiftChange shiftChange
= iterator
.next();
325 result
[pos
] = new SingleReadAlignment
.Shift(shiftChange
.getBaseCallIndex(), shiftChange
.getShiftChange());
332 * @see org.eclipse.ui.part.EditorPart#doSave(org.eclipse.core.runtime.IProgressMonitor)
335 public void doSave(IProgressMonitor monitor
) {
336 if (getEditorInput() instanceof AlignmentEditorInput
) {
337 String taskName
= "Saving alignment";
338 monitor
.beginTask(taskName
, 3);
340 Sequence sequenceNode
= ((AlignmentEditorInput
)getEditorInput()).getSequenceNode();
341 StringAdapter stringProvider
= new StringAdapter(getConsensusArea().getSequenceProvider(), false); // Throws an exception if a token has more than one character.
343 // Write consensus sequence:
344 SequenceString consensusSequenceObj
= sequenceNode
.getConsensusSequence();
345 String newConsensusSequence
= stringProvider
.getSequence(
346 getConsensusArea().getSequenceProvider().sequenceIDByName(CONSENSUS_NAME
));
347 if (consensusSequenceObj
== null) {
348 sequenceNode
.setConsensusSequence(SequenceString
.NewInstance(newConsensusSequence
));
351 consensusSequenceObj
.setString(newConsensusSequence
);
354 // Write single reads:
355 stringProvider
.setUnderlyingProvider(getReadsArea().getSequenceProvider());
356 sequenceNode
.getSingleReadAlignments().retainAll(cdmMap
.values()); // Remove all reads that are not in the alignment anymore.
357 Iterator
<Integer
> iterator
= getReadsArea().getSequenceProvider().sequenceIDIterator();
358 while (iterator
.hasNext()) {
359 int id
= iterator
.next();
360 SingleReadAlignment singleRead
= cdmMap
.get(id
);
361 if (singleRead
== null) {
362 //TODO Create new read object. => Shall it be allowed to add reads in the alignment editor which are not represented in the CDM tree before the alignment editor is saved?
363 //singleRead = SingleReadAlignment.NewInstance(consensusSequence, singleRead, shifts, editedSequence);
366 singleRead
.setEditedSequence(stringProvider
.getSequence(id
));
367 singleRead
.setReverseComplement(getPherogramArea(id
).getProvider() instanceof ReverseComplementPherogramProvider
); // Works only if ReverseComplementPherogramProvider instances are not nested.
368 singleRead
.setShifts(convertToCDMShifts(getPherogramArea(id
).getAlignmentModel()));
371 if (!conversationHolder
.isBound()) {
372 conversationHolder
.bind();
376 // Commit the conversation and start a new transaction immediately:
377 conversationHolder
.commit(true);
383 firePropertyChange(PROP_DIRTY
);
386 //TODO Throw exception as soon as testing period which allows unlinked AlignmentEditor is over.
392 * @see org.eclipse.ui.part.EditorPart#doSaveAs()
395 public void doSaveAs() {}
399 * @see org.eclipse.ui.part.EditorPart#init(org.eclipse.ui.IEditorSite, org.eclipse.ui.IEditorInput)
402 public void init(IEditorSite site
, IEditorInput input
) throws PartInitException
{
409 * @see org.eclipse.ui.part.EditorPart#isDirty()
412 public boolean isDirty() {
417 private void setDirty() {
419 firePropertyChange(IEditorPart
.PROP_DIRTY
);
424 * @see org.eclipse.ui.part.EditorPart#isSaveAsAllowed()
427 public boolean isSaveAsAllowed() {
428 return false; // "Save as" not allowed.
433 public void setFocus() {
434 if(conversationHolder
!=null){
435 conversationHolder
.bind();
440 public boolean isInsertMode() {
441 return getAlignmentsContainer().getEditSettings().isInsert();
445 public boolean isInsertLeftInPherogram() {
446 return getAlignmentsContainer().getEditSettings().isInsertLeftInDataArea();
450 public void toggleLeftRightInsertionInPherogram() {
451 getAlignmentsContainer().getEditSettings().toggleInsertLeftInDataArea();
455 public void toggleInsertOverwrite() {
456 getAlignmentsContainer().getEditSettings().toggleInsert();
460 public void reverseComplementSelection() {
461 SelectionModel selection
= getReadsArea().getSelection();
462 SequenceDataProvider
<?
> provider
= getReadsArea().getSequenceProvider();
463 for (int row
= selection
.getStartRow(); row
< selection
.getStartRow() + selection
.getCursorHeight(); row
++) {
464 int sequenceID
= getReadsArea().getSequenceOrder().idByIndex(row
);
465 //TODO rc edited sequence
467 if (getPherogramArea(sequenceID
).getProvider() instanceof ReverseComplementPherogramProvider
) {
468 //getPherogramArea(sequenceID).
469 //TODO Allow to set new provider in PherogramArea or create new PherogramArea
470 //TODO Reposition pherogram according to previous position in edited sequence and length
476 public static PherogramProvider
readPherogram(URI uri
) throws IOException
, UnsupportedChromatogramFormatException
{
477 PherogramProvider result
;
478 InputStream stream
= uri
.toURL().openStream();
480 result
= new BioJavaPherogramProvider(ChromatogramFactory
.create(stream
));
489 private String
newReadName() {
491 while (getReadsArea().getSequenceProvider().sequenceIDByName(DEFAULT_READ_NAME_PREFIX
+ index
)
492 != SequenceDataProvider
.NO_SEQUENCE_FOUND
) {
496 return DEFAULT_READ_NAME_PREFIX
+ index
;
500 public void addRead(URI pherogramURI
, boolean reverseComplemented
) throws IOException
, UnsupportedChromatogramFormatException
{
501 addRead(newReadName(), pherogramURI
, reverseComplemented
, null, null);
506 * Adds a new sequence with attached phergram data area to the reads alignment.
508 * If {@code null} is specified as {@code editedSequence} the base call sequence from the pherogram will
509 * be set as the edited sequence. If {@code null} is specified as {@code shifts} no shifts between the edited
510 * and the base calls sequence are assumed.
512 * @param name the name of the new sequence
513 * @param pherogramURI the URI where the associated pherogram file is located
514 * @param reverseComplemented Specify {@code true} here, if the reverse complement of the pherogram data should
515 * be added, {@code false} otherwise.
516 * @param editedSequence the edited version of the base call sequence (May be {@code null}.)
517 * @param shifts the alignment information that links the edited and the base call sequence (May be {@code null}.)
518 * @return the sequence ID of the added read
519 * @throws IOException if an error occurred when trying to read the pherogram file
520 * @throws UnsupportedChromatogramFormatException if the format of the pherogram file is not supported
522 public int addRead(String name
, URI pherogramURI
, boolean reverseComplemented
, String editedSequence
,
523 SingleReadAlignment
.Shift
[] shifts
) throws IOException
, UnsupportedChromatogramFormatException
{
525 SequenceDataProvider provider
= getReadsArea().getSequenceProvider();
526 PherogramProvider pherogramProvider
= readPherogram(pherogramURI
); // Must happen before a sequence is added, because it might throw an exception.
527 if (reverseComplemented
) {
528 pherogramProvider
= new ReverseComplementPherogramProvider(pherogramProvider
);
532 provider
.addSequence(name
);
533 int id
= provider
.sequenceIDByName(name
);
535 // Set edited sequence:
536 Collection
<Object
> tokens
; // First save tokens in a collection to avoid GUI updated for each token.
537 if (editedSequence
!= null) {
538 tokens
= SequenceUtils
.stringToTokenList(editedSequence
, provider
.getTokenSet());
540 else { // Copy base call sequence into alignment:
541 tokens
= new ArrayList
<Object
>();
542 for (int i
= 0; i
< pherogramProvider
.getSequenceLength(); i
++) {
543 tokens
.add(provider
.getTokenSet().tokenByKeyChar(
544 pherogramProvider
.getBaseCall(i
).getUpperedBase().charAt(0)));
548 provider
.insertTokensAt(id
, 0, tokens
);
550 // Create pherogram area:
551 PherogramArea pherogramArea
= new PherogramArea(getReadsArea().getContentArea(), pherogramProvider
);
554 if ((shifts
!= null) && (shifts
.length
> 0)) {
555 PherogramAlignmentModel alignmentModel
= pherogramArea
.getAlignmentModel();
556 for (int i
= 0; i
< shifts
.length
; i
++) {
557 alignmentModel
.addShiftChange(shifts
[i
].position
, shifts
[i
].shift
);
562 // Add pherogram area to GUI:
563 pherogramArea
.addMouseListener(new PherogramMouseListener(pherogramURI
));
564 getReadsArea().getDataAreas().getSequenceAreas(id
).add(pherogramArea
);