Test version of an AlignmentArea added to AlignmentEditor.
[taxeditor.git] / eu.etaxonomy.taxeditor.store / src / main / java / eu / etaxonomy / taxeditor / ui / selection / EntitySelectionElement.java
1 /**
2 *
3 */
4 package eu.etaxonomy.taxeditor.ui.selection;
5
6 import java.util.EnumSet;
7 import java.util.Observable;
8 import java.util.Observer;
9
10 import org.eclipse.core.runtime.IStatus;
11 import org.eclipse.jface.wizard.WizardDialog;
12 import org.eclipse.swt.SWT;
13 import org.eclipse.swt.events.SelectionAdapter;
14 import org.eclipse.swt.events.SelectionEvent;
15 import org.eclipse.swt.events.SelectionListener;
16 import org.eclipse.swt.graphics.Color;
17 import org.eclipse.swt.widgets.Button;
18 import org.eclipse.swt.widgets.Composite;
19 import org.eclipse.swt.widgets.Label;
20 import org.eclipse.swt.widgets.Shell;
21 import org.eclipse.swt.widgets.Text;
22 import org.springframework.security.core.GrantedAuthority;
23
24 import eu.etaxonomy.cdm.api.conversation.ConversationHolder;
25 import eu.etaxonomy.cdm.api.conversation.IConversationEnabled;
26 import eu.etaxonomy.cdm.common.CdmUtils;
27 import eu.etaxonomy.cdm.model.common.CdmBase;
28 import eu.etaxonomy.cdm.model.common.Group;
29 import eu.etaxonomy.cdm.model.common.ICdmBase;
30 import eu.etaxonomy.cdm.model.common.IIdentifiableEntity;
31 import eu.etaxonomy.cdm.model.common.User;
32 import eu.etaxonomy.cdm.model.molecular.Amplification;
33 import eu.etaxonomy.cdm.model.molecular.Primer;
34 import eu.etaxonomy.cdm.persistence.hibernate.CdmDataChangeMap;
35 import eu.etaxonomy.cdm.persistence.hibernate.permission.CRUD;
36 import eu.etaxonomy.taxeditor.model.ImageResources;
37 import eu.etaxonomy.taxeditor.preference.Resources;
38 import eu.etaxonomy.taxeditor.store.CdmStore;
39 import eu.etaxonomy.taxeditor.store.LoginManager;
40 import eu.etaxonomy.taxeditor.store.StoreUtil;
41 import eu.etaxonomy.taxeditor.ui.dialog.selection.SelectionDialogFactory;
42 import eu.etaxonomy.taxeditor.ui.element.AbstractCdmFormElement;
43 import eu.etaxonomy.taxeditor.ui.element.CdmFormFactory;
44 import eu.etaxonomy.taxeditor.ui.element.CdmPropertyChangeEvent;
45 import eu.etaxonomy.taxeditor.ui.element.ICdmFormElement;
46 import eu.etaxonomy.taxeditor.ui.element.IEnableableFormElement;
47 import eu.etaxonomy.taxeditor.ui.element.IEntityElement;
48 import eu.etaxonomy.taxeditor.ui.element.ILabeledElement;
49 import eu.etaxonomy.taxeditor.ui.element.ISelectableElement;
50 import eu.etaxonomy.taxeditor.ui.element.LayoutConstants;
51 import eu.etaxonomy.taxeditor.ui.element.SelectionArbitrator;
52 import eu.etaxonomy.taxeditor.ui.section.grantedAuthority.GrantedAuthorityLabelTextProvider;
53
54 /**
55 * <p>
56 * Abstract AbstractSelectionElement class.
57 * </p>
58 *
59 * @author n.hoffmann
60 * @created Nov 17, 2009
61 * @version 1.0
62 * @param <T>
63 */
64 public class EntitySelectionElement<T extends ICdmBase> extends
65 AbstractCdmFormElement implements SelectionListener, IEnableableFormElement, ISelectableElement, IEntityElement<T>, ILabeledElement, IConversationEnabled, Observer {
66
67 private static final EnumSet<CRUD> UPDATE = EnumSet.of(CRUD.UPDATE);
68 private static final EnumSet<CRUD> DELETE = EnumSet.of(CRUD.DELETE);
69 private static final EnumSet<CRUD> CREATE = EnumSet.of(CRUD.CREATE);
70
71 /**
72 * Bitmask for configuring functionality of selection element
73 */
74 public static final int NOTHING = 0; // 000
75 public static final int EDITABLE = 1 << 0; // 001
76 public static final int DELETABLE = 1 << 1; // 010
77 public static final int SELECTABLE = 1 << 2; // 100
78 public static final int ALL = EDITABLE | DELETABLE | SELECTABLE; // 111
79
80 protected T entity;
81
82 protected Label label;
83 protected Text text;
84 protected Button button_selection;
85
86 private SelectionArbitrator selectionArbitrator;
87
88 protected Button button_edit;
89
90 private final String labelString;
91
92 private Composite selectableComposite;
93
94 private Button button_remove;
95
96 private final boolean isEditable;
97
98 private final boolean isDeletable;
99
100 private final ConversationHolder conversation;
101 private Class<T> clazz;
102
103 /**
104 * <p>
105 * Constructor for AbstractSelectionElement.
106 * </p>
107 *
108 * @param formFactory
109 * a {@link eu.etaxonomy.taxeditor.ui.element.CdmFormFactory}
110 * object.
111 * @param conversation
112 * TODO
113 * @param parentElement
114 * a {@link eu.etaxonomy.taxeditor.ui.element.ICdmFormElement}
115 * object.
116 * @param labelString
117 * a {@link java.lang.String} object.
118 * @param entity
119 * a T object.
120 * @param isEditable
121 * a boolean.
122 * @param isSelectable
123 * a boolean.
124 * @param isDeletable
125 * a boolean.
126 * @param style
127 * a int.
128 * @param <T>
129 * a T object.
130 */
131 public EntitySelectionElement(CdmFormFactory formFactory,
132 ConversationHolder conversation, ICdmFormElement parentElement,
133 String labelString, T entity, int mode, int style) {
134 super(formFactory, parentElement);
135
136 this.isEditable = (mode & EDITABLE) == EDITABLE;
137 this.isDeletable = (mode & DELETABLE) == DELETABLE;
138 boolean isSelectable = (mode & SELECTABLE) == SELECTABLE;
139
140 this.labelString = labelString;
141
142 this.conversation = conversation;
143
144 if (isSelectable && formFactory.getSelectionProvider() != null) {
145 selectionArbitrator = formFactory.createSelectionArbitrator(this);
146 }
147
148 createControls(getLayoutComposite(), SWT.NULL);
149
150 setEntity(entity);
151 }
152
153 public EntitySelectionElement(CdmFormFactory formFactory,
154 ConversationHolder conversation, ICdmFormElement parentElement, Class<T> clazz,
155 String labelString, T entity, int mode, int style) {
156 this(formFactory, conversation, parentElement, labelString, entity, mode, style);
157 this.clazz = clazz;
158 }
159
160 private void createControls(Composite parent, int style) {
161
162 label = formFactory.createLabel(getLayoutComposite(), labelString + " : ",
163 SWT.NULL);
164
165 addControl(label);
166
167 selectableComposite = formFactory.createComposite(getLayoutComposite());
168
169 int columns = 2;
170 if (isEditable) {
171 columns += 1;
172 }
173 if (isDeletable) {
174 columns += 1;
175 }
176
177 selectableComposite.setLayout(LayoutConstants.LAYOUT(columns, false));
178 selectableComposite.setLayoutData(LayoutConstants.FILL_HORIZONTALLY());
179
180 addControl(selectableComposite);
181
182 text = formFactory.createText(selectableComposite, null, SWT.WRAP);
183 text.setEditable(false);
184 addControl(text);
185
186 text.setLayoutData(LayoutConstants.FILL_HORIZONTALLY());
187 text.setBackground(StoreUtil
188 .getColor(Resources.COLOR_TEXT_DISABLED_BACKGROUND));
189
190 button_selection = formFactory.createButton(selectableComposite, null,
191 SWT.PUSH);
192 button_selection.setImage(ImageResources
193 .getImage(ImageResources.BROWSE_ICON));
194 button_selection.setToolTipText("Browse existing");
195
196 addControl(button_selection);
197 button_selection.addSelectionListener(this);
198
199 if (isEditable) {
200 button_edit = formFactory.createButton(selectableComposite, null,
201 SWT.PUSH);
202 button_edit.setImage(ImageResources
203 .getImage(ImageResources.EDIT_ICON));
204 button_edit.setToolTipText("Edit");
205 addControl(button_edit);
206 button_edit.addSelectionListener(new EditListener(this));
207 }
208
209 if (isDeletable) {
210 button_remove = formFactory.createButton(selectableComposite, null,
211 SWT.PUSH);
212 button_remove.setImage(ImageResources
213 .getImage(ImageResources.TRASH_ICON));
214 button_remove.setToolTipText("Remove");
215 addControl(button_remove);
216 button_remove.addSelectionListener(new DeleteListener(this));
217 }
218 }
219
220 @Override
221 public void widgetSelected(SelectionEvent e) {
222 T selection = SelectionDialogFactory.getSelectionFromDialog(clazz, getShell(), getConversationHolder(), getEntity());
223 setSelectionInternal(selection);
224 }
225
226 /**
227 * Return the selected object
228 *
229 * @return a T object.
230 */
231 public T getSelection() {
232 return entity;
233 }
234
235 /*
236 * (non-Javadoc)
237 *
238 * @see
239 * eu.etaxonomy.taxeditor.forms.IEnableableFormElement#setEnabled(boolean)
240 */
241 /** {@inheritDoc} */
242 @Override
243 public void setEnabled(boolean enabled) {
244 button_selection.setEnabled(enabled);
245 if (isEditable) {
246 updateButtonStates();
247 }
248 }
249
250 /* (non-Javadoc)
251 * @see eu.etaxonomy.taxeditor.ui.element.IEnableableFormElement#isEnabled()
252 */
253 @Override
254 public boolean isEnabled() {
255 return button_selection.isEnabled();
256 }
257
258 /**
259 * <p>
260 * setSelectionInternal
261 * </p>
262 *
263 * @param selection
264 * a T object.
265 */
266 protected void setSelectionInternal(T selection) {
267 if (selection != null && !selection.equals(this.entity)) {
268 setEntity(selection);
269 firePropertyChangeEvent(new CdmPropertyChangeEvent(this, null));
270 }
271 }
272
273 /**
274 * <p>
275 * Setter for the field <code>entity</code>.
276 * </p>
277 *
278 * @param selection
279 * a T object.
280 */
281 public void setEntity(T selection) {
282 this.entity = selection;
283 updateElement();
284 }
285
286 /**
287 * Updates this elements view
288 */
289 protected void updateElement() {
290 String title = CdmUtils.Nz(getTitle());
291 // we have to duplicate ampersands otherwise they are treated as
292 // mnenomic (see Label.setText() documentation)
293 title = title.replace("&", "&&");
294 text.setText(title); // title can be null
295 if (isEditable) {
296 updateButtonStates();
297 }
298 }
299
300 public void updateFromWizard() {
301 updateElement();
302 firePropertyChangeEvent(new CdmPropertyChangeEvent(this, null));
303 }
304
305 /**
306 * <p>
307 * getTitle
308 * </p>
309 *
310 * @return a {@link java.lang.String} object.
311 */
312 protected String getTitle() {
313 if (entity != null){
314 if(entity instanceof IIdentifiableEntity) {
315 return ((IIdentifiableEntity) entity).getTitleCache();
316 } else if(entity instanceof Group){
317 return ((Group) entity).getName();
318 } else if(entity instanceof GrantedAuthority){
319 return GrantedAuthorityLabelTextProvider.getText(((GrantedAuthority) entity));
320 } else if(entity instanceof User){
321 return ((User) entity).getUsername();
322 } else if (entity instanceof Primer){
323 return ((Primer) entity).getLabel();
324 } else if (entity instanceof Amplification){
325 return ((Amplification) entity).getDescription();
326 }
327
328 }
329 return "";
330 }
331
332 /** {@inheritDoc} */
333 @Override
334 public void setSelected(boolean selected) {
335 setBackground(selected ? SELECTED : getPersistentBackground());
336 }
337
338 /*
339 * (non-Javadoc)
340 *
341 * @see eu.etaxonomy.taxeditor.forms.IEntityElement#getEntity()
342 */
343 /**
344 * <p>
345 * Getter for the field <code>entity</code>.
346 * </p>
347 *
348 * @return a T object.
349 */
350 @Override
351 public T getEntity() {
352 return entity;
353 }
354
355 /*
356 * (non-Javadoc)
357 *
358 * @see eu.etaxonomy.taxeditor.forms.section.cdmdetail.ISelectableElement#
359 * getSelectionArbitrator()
360 */
361 /**
362 * <p>
363 * Getter for the field <code>selectionArbitrator</code>.
364 * </p>
365 *
366 * @return a {@link eu.etaxonomy.taxeditor.ui.element.SelectionArbitrator}
367 * object.
368 */
369 @Override
370 public SelectionArbitrator getSelectionArbitrator() {
371 return selectionArbitrator;
372 }
373
374 /**
375 * Convenient access to current shell
376 *
377 * @return a {@link org.eclipse.swt.widgets.Shell} object.
378 */
379 public Shell getShell() {
380 return getLayoutComposite().getShell();
381 }
382
383 /** {@inheritDoc} */
384 @Override
385 public void setIrrelevant(boolean irrelevant) {
386 String colorId = irrelevant ? Resources.COLOR_COMPOSITE_IRRELEVANT
387 : Resources.COLOR_TEXT_DISABLED_BACKGROUND;
388
389 Color color = StoreUtil.getColor(colorId);
390 text.setBackground(color);
391 }
392
393 private class DeleteListener extends SelectionAdapter {
394
395 private final EntitySelectionElement<T> selectionElement;
396
397 public DeleteListener(EntitySelectionElement<T> selectionElement) {
398 this.selectionElement = selectionElement;
399 }
400
401 @Override
402 public void widgetSelected(SelectionEvent e) {
403 setEntity(null);
404 firePropertyChangeEvent(new CdmPropertyChangeEvent(
405 selectionElement, null));
406 }
407 }
408
409 private class EditListener extends SelectionAdapter {
410
411 private final EntitySelectionElement<T> selectionElement;
412
413 public EditListener(EntitySelectionElement<T> selectionElement) {
414 this.selectionElement = selectionElement;
415 }
416
417 /** {@inheritDoc} */
418 @Override
419 public void widgetSelected(SelectionEvent e) {
420 WizardDialog dialog = new WizardDialog(selectionElement.getShell(),
421 new EditFromSelectionWizard(selectionElement));
422 if (dialog.open() == IStatus.OK) {
423 selectionElement.updateFromWizard();
424 }
425 }
426 }
427
428 // not used
429 /** {@inheritDoc} */
430 @Override
431 public void widgetDefaultSelected(SelectionEvent e) {
432 }
433
434 /**
435 * <p>
436 * getConversationHolder
437 * </p>
438 *
439 * @return a {@link eu.etaxonomy.cdm.api.conversation.ConversationHolder}
440 * object.
441 */
442 @Override
443 public ConversationHolder getConversationHolder() {
444 return conversation;
445 }
446
447 /** {@inheritDoc} */
448 @Override
449 public void setBackground(Color color) {
450 if(!label.isDisposed()){
451 label.setBackground(color);
452 }
453 }
454
455 /** {@inheritDoc} */
456 @Override
457 public void setLabel(String labelString) {
458 if (label != null) {
459 label.setText(labelString);
460 }
461 }
462
463 /**
464 * <p>
465 * Getter for the field <code>label</code>.
466 * </p>
467 *
468 * @return a {@link java.lang.String} object.
469 */
470 @Override
471 public String getLabel() {
472 if (label != null) {
473 return label.getText() + " : ";
474 }
475 return null;
476 }
477
478 /** {@inheritDoc} */
479 @Override
480 public void update(CdmDataChangeMap changeEvents) {
481 }
482
483 /* (non-Javadoc)
484 * @see eu.etaxonomy.taxeditor.ui.element.AbstractCdmFormElement#removeElements()
485 */
486 @Override
487 public void removeElements(){
488 super.removeElements();
489 LoginManager loginManager = CdmStore.getLoginManager();
490 loginManager.addObserver(this);
491 }
492
493 @Override
494 public void update(Observable o, Object arg) {
495 if(o instanceof LoginManager){
496 updateButtonStates();
497 }
498 }
499
500 private void updateButtonStates() {
501 if(button_edit != null && !button_selection.isDisposed()){
502 button_edit.setEnabled(isEditable && button_selection.isEnabled() && getEntity() != null && CdmStore.currentAuthentiationHasPermission((CdmBase) getEntity(), UPDATE));
503 }
504 }
505 }