Project

General

Profile

Download (19.4 KB) Statistics
| Branch: | Tag: | Revision:
1
/**
2
 * Copyright (C) 2007 EDIT
3
 * European Distributed Institute of Taxonomy
4
 * http://www.e-taxonomy.eu
5
 *
6
 * The contents of this file are subject to the Mozilla Public License Version 1.1
7
 * See LICENSE.TXT at the top of this package for the full license terms.
8
 */
9

    
10
package eu.etaxonomy.taxeditor.navigation.navigator.e4;
11

    
12
import java.util.ArrayList;
13
import java.util.Arrays;
14
import java.util.Comparator;
15
import java.util.HashMap;
16
import java.util.HashSet;
17
import java.util.List;
18
import java.util.Map;
19
import java.util.Observable;
20
import java.util.Observer;
21
import java.util.Set;
22
import java.util.UUID;
23

    
24
import javax.annotation.PostConstruct;
25
import javax.annotation.PreDestroy;
26
import javax.inject.Inject;
27

    
28
import org.eclipse.core.commands.operations.UndoContext;
29
import org.eclipse.core.runtime.IAdaptable;
30
import org.eclipse.core.runtime.IProgressMonitor;
31
import org.eclipse.e4.core.contexts.ContextInjectionFactory;
32
import org.eclipse.e4.core.contexts.IEclipseContext;
33
import org.eclipse.e4.core.di.annotations.Optional;
34
import org.eclipse.e4.ui.di.Focus;
35
import org.eclipse.e4.ui.di.UIEventTopic;
36
import org.eclipse.e4.ui.di.UISynchronize;
37
import org.eclipse.e4.ui.model.application.MApplication;
38
import org.eclipse.e4.ui.model.application.ui.basic.MPart;
39
import org.eclipse.e4.ui.model.application.ui.menu.MToolBar;
40
import org.eclipse.e4.ui.model.application.ui.menu.MToolBarElement;
41
import org.eclipse.e4.ui.model.application.ui.menu.impl.HandledToolItemImpl;
42
import org.eclipse.e4.ui.services.EMenuService;
43
import org.eclipse.e4.ui.workbench.modeling.EModelService;
44
import org.eclipse.e4.ui.workbench.modeling.EPartService;
45
import org.eclipse.e4.ui.workbench.modeling.ESelectionService;
46
import org.eclipse.jface.util.LocalSelectionTransfer;
47
import org.eclipse.jface.viewers.IElementComparer;
48
import org.eclipse.jface.viewers.ISelection;
49
import org.eclipse.jface.viewers.ISelectionChangedListener;
50
import org.eclipse.jface.viewers.IStructuredSelection;
51
import org.eclipse.jface.viewers.StructuredSelection;
52
import org.eclipse.jface.viewers.TreePath;
53
import org.eclipse.jface.viewers.TreeViewer;
54
import org.eclipse.swt.SWT;
55
import org.eclipse.swt.dnd.DND;
56
import org.eclipse.swt.dnd.Transfer;
57
import org.eclipse.swt.layout.FillLayout;
58
import org.eclipse.swt.widgets.Composite;
59
import org.eclipse.swt.widgets.Tree;
60
import org.eclipse.ui.IMemento;
61

    
62
import eu.etaxonomy.cdm.api.application.CdmApplicationState;
63
import eu.etaxonomy.cdm.api.application.CdmChangeEvent;
64
import eu.etaxonomy.cdm.api.application.CdmChangeEvent.Action;
65
import eu.etaxonomy.cdm.api.application.ICdmChangeListener;
66
import eu.etaxonomy.cdm.api.conversation.ConversationHolder;
67
import eu.etaxonomy.cdm.api.conversation.IConversationEnabled;
68
import eu.etaxonomy.cdm.api.service.IClassificationService;
69
import eu.etaxonomy.cdm.api.service.ITaxonNodeService;
70
import eu.etaxonomy.cdm.model.common.CdmBase;
71
import eu.etaxonomy.cdm.model.common.ICdmBase;
72
import eu.etaxonomy.cdm.model.taxon.Classification;
73
import eu.etaxonomy.cdm.model.taxon.TaxonNode;
74
import eu.etaxonomy.cdm.persistence.dto.TaxonNodeDto;
75
import eu.etaxonomy.cdm.persistence.dto.TaxonNodeDtoByNameComparator;
76
import eu.etaxonomy.cdm.persistence.dto.TaxonNodeDtoByRankAndNameComparator;
77
import eu.etaxonomy.cdm.persistence.dto.TaxonNodeDtoNaturalComparator;
78
import eu.etaxonomy.cdm.persistence.hibernate.CdmDataChangeMap;
79
import eu.etaxonomy.taxeditor.editor.ITaxonEditor;
80
import eu.etaxonomy.taxeditor.editor.name.e4.TaxonNameEditorE4;
81
import eu.etaxonomy.taxeditor.event.WorkbenchEventConstants;
82
import eu.etaxonomy.taxeditor.model.AbstractUtility;
83
import eu.etaxonomy.taxeditor.model.DataChangeBridge;
84
import eu.etaxonomy.taxeditor.model.IContextListener;
85
import eu.etaxonomy.taxeditor.model.IDataChangeBehavior;
86
import eu.etaxonomy.taxeditor.navigation.AppModelId;
87
import eu.etaxonomy.taxeditor.navigation.NavigationUtil;
88
import eu.etaxonomy.taxeditor.navigation.l10n.Messages;
89
import eu.etaxonomy.taxeditor.navigation.navigator.EmptyRoot;
90
import eu.etaxonomy.taxeditor.navigation.navigator.Root;
91
import eu.etaxonomy.taxeditor.navigation.navigator.TaxonNodeNavigatorComparator;
92
import eu.etaxonomy.taxeditor.operation.IPostOperationEnabled;
93
import eu.etaxonomy.taxeditor.preference.NavigatorOrderEnum;
94
import eu.etaxonomy.taxeditor.preference.PreferencesUtil;
95
import eu.etaxonomy.taxeditor.session.ICdmEntitySession;
96
import eu.etaxonomy.taxeditor.session.ICdmEntitySessionEnabled;
97
import eu.etaxonomy.taxeditor.store.CdmStore;
98
import eu.etaxonomy.taxeditor.store.LoginManager;
99
import eu.etaxonomy.taxeditor.ui.element.LayoutConstants;
100
import eu.etaxonomy.taxeditor.workbench.part.ICollapsableExpandable;
101

    
102
/**
103
 *
104
 * @author pplitzner
105
 * @since Sep 7, 2017
106
 *
107
 */
108
public class TaxonNavigatorE4 implements
109
		IPostOperationEnabled, IConversationEnabled, Observer,
110
		ICdmEntitySessionEnabled, ICdmChangeListener, IContextListener,
111
		ICollapsableExpandable {
112

    
113
    private static final String RESTORING_TAXON_NAVIGATOR = Messages.TaxonNavigator_RESTORE;
114

    
115
	private static final String TREE_PATH = "treepath"; //$NON-NLS-1$
116

    
117
	private static final String TREE_PATHS = "treepaths"; //$NON-NLS-1$
118

    
119
	private final int dndOperations = DND.DROP_MOVE;
120

    
121
	private ConversationHolder conversation;
122

    
123
	private ICdmEntitySession cdmEntitySession;
124

    
125
	private IDataChangeBehavior dataChangeBehavior;
126

    
127
	private Root root;
128

    
129
	private TreeViewer viewer;
130

    
131
    @Inject
132
    private ESelectionService selService;
133

    
134
    @Inject
135
    private UISynchronize sync;
136

    
137
    private ISelectionChangedListener selectionChangedListener;
138

    
139
    private UndoContext undoContext;
140

    
141
    @Inject
142
    private MApplication application;
143

    
144
    @Inject
145
    private EModelService modelService;
146

    
147
    @Inject
148
    private EPartService partService;
149

    
150
    private boolean linkWithTaxon = false;
151

    
152
    @Inject
153
    public TaxonNavigatorE4() {
154
	    undoContext = new UndoContext();
155
	    CdmStore.getContextManager().addContextListener(this);
156
    }
157

    
158
	@PostConstruct
159
	private void create(Composite parent, EMenuService menuService, IEclipseContext context){
160
	    FillLayout layout = new FillLayout();
161
	    layout.marginHeight = 0;
162
	    layout.marginWidth = 0;
163
	    layout.type = SWT.VERTICAL;
164

    
165
	    parent.setLayout(layout);
166
	    viewer = new TreeViewer(new Tree(parent, SWT.H_SCROLL | SWT.V_SCROLL | SWT.FULL_SELECTION | SWT.MULTI));
167
	    viewer.getControl().setLayoutData(LayoutConstants.FILL());
168

    
169
	    viewer.setContentProvider(new TaxonNavigatorContentProviderE4());
170
        viewer.setLabelProvider(new TaxonNavigatorLabelProviderE4());
171
        viewer.addDoubleClickListener(event->{
172
            ISelection selection = event.getSelection();
173
            if(selection instanceof IStructuredSelection){
174
                Object firstElement = ((IStructuredSelection) selection).getFirstElement();
175
                if(firstElement instanceof ICdmBase){
176
                    NavigationUtil.openEditor((ICdmBase) firstElement, viewer.getControl().getShell(), modelService, partService, application);
177
                }
178
                if(firstElement instanceof TaxonNodeDto){
179
                    NavigationUtil.openEditor((TaxonNodeDto) firstElement, viewer.getControl().getShell(), modelService, partService, application);
180
                }
181
            }
182
        });
183

    
184
        //propagate selection
185
        selectionChangedListener = (event -> selService.setSelection(event.getSelection()));
186
        viewer.addSelectionChangedListener(selectionChangedListener);
187

    
188
        //create context menu
189
        menuService.registerContextMenu(viewer.getControl(), AppModelId.POPUPMENU_EU_ETAXONOMY_TAXEDITOR_NAVIGATOR_POPUPMENU_TAXONNAVIGATOR );
190

    
191
        //add drag'n'drop support
192
        Transfer[] transfers = new Transfer[] {LocalSelectionTransfer.getTransfer()};
193
        viewer.addDragSupport(dndOperations, transfers, new TreeNodeDragListenerE4(viewer));
194
        TreeNodeDropAdapterE4 dropAdapter = new TreeNodeDropAdapterE4(this);
195
        ContextInjectionFactory.inject(dropAdapter, context);
196
        viewer.addDropSupport(dndOperations, transfers, dropAdapter);
197

    
198
        updateSyncButton();
199

    
200

    
201

    
202
	    init();
203
	}
204

    
205
    protected void updateSyncButton() {
206
        MPart viewPart = partService.findPart(AppModelId.PARTDESCRIPTOR_EU_ETAXONOMY_TAXEDITOR_NAVIGATION_NAVIGATOR);
207
        if(viewPart!=null){
208
            MToolBar toolBar = viewPart.getToolbar();
209

    
210
            List<MToolBarElement> toolBarElements = toolBar.getChildren();
211

    
212
            MToolBarElement upperHandledMenuItem = toolBarElements.get(1);
213
            if (upperHandledMenuItem instanceof HandledToolItemImpl){
214
                ((HandledToolItemImpl)upperHandledMenuItem).setSelected(linkWithTaxon);
215
            }
216
        }
217
    }
218

    
219
	/** {@inheritDoc} */
220
	protected IAdaptable getInitialInput() {
221
		Comparator<TaxonNodeDto> comparator;
222
		NavigatorOrderEnum orderValue = PreferencesUtil.getSortNodes();
223
		if (orderValue.equals(NavigatorOrderEnum.NaturalOrder)){
224
			comparator = new TaxonNodeDtoNaturalComparator();
225
		} else if (orderValue.equals(NavigatorOrderEnum.AlphabeticalOrder)){
226
			comparator = new TaxonNodeDtoByNameComparator();
227
		}else {
228
			comparator = new TaxonNodeDtoByRankAndNameComparator();
229
		}
230

    
231
		TaxonNodeNavigatorComparator viewerComparator = new TaxonNodeNavigatorComparator(comparator);
232
		viewer.setComparator(viewerComparator);
233
		viewer.setComparer(new IElementComparer() {
234

    
235
	        @Override
236
            public int hashCode(Object element) {
237
	            if (element instanceof TaxonNodeDto){
238
    	            TaxonNodeDto nodeDto = (TaxonNodeDto)element;
239

    
240
    	            String s = nodeDto.getUuid().toString();
241
    	            if (s != null) {
242
    	                return s.hashCode();
243
    	            }
244
    	            return element.hashCode();
245
	            }else{
246
	                return element.toString().hashCode();
247
	            }
248
	        }
249

    
250
            @Override
251
            public boolean equals(Object element1, Object element2) {
252
                if (element1 instanceof TaxonNodeDto && element2 instanceof TaxonNodeDto){
253
                    TaxonNodeDto node1 = (TaxonNodeDto)element1;
254
                    TaxonNodeDto node2 = (TaxonNodeDto)element2;
255
                    return (node1.getUuid().equals(node2.getUuid()));
256
                }else {
257
                    return element1.equals(element2);
258
                }
259
            }
260
		}
261
		);
262

    
263
		if (CdmStore.isActive()) {
264

    
265
			// TODO when closing and reopening the taxon navigator
266
			// we do not preserve state. Closing the view, in contrary to
267
			// closing the whole application
268
			// should be handled by the state manager too
269
		    root = new Root(conversation);
270

    
271
			return root;
272
		}
273
		return new EmptyRoot();
274
	}
275

    
276
	public void init() {
277
		if (CdmStore.isActive() && conversation == null) {
278
			conversation = CdmStore.createConversation();
279
			conversation.registerForDataStoreChanges(TaxonNavigatorE4.this);
280
		}
281
		if (CdmStore.isActive()) {
282
		    cdmEntitySession = CdmStore.getCurrentSessionManager().newSession(this, true);
283
		    CdmApplicationState.getCurrentDataChangeService().register(this);
284
		    viewer.setInput(getInitialInput());
285
		}
286
		CdmStore.getLoginManager().addObserver(this);
287

    
288
	}
289

    
290
	//Link with taxon selection
291
	@Inject
292
	@Optional
293
	public void updateCurrentTaxon(@UIEventTopic(WorkbenchEventConstants.CURRENT_ACTIVE_EDITOR)ITaxonEditor editor){
294
	    if(linkWithTaxon && editor!=null){
295
	        viewer.refresh();
296
	        TaxonNodeDto taxonNode = null;
297
	        if(editor.getTaxon()!=null && editor.getTaxon().getTaxonNodes()!=null){
298
	            if (editor instanceof TaxonNameEditorE4){
299
	                taxonNode = new TaxonNodeDto( ((TaxonNameEditorE4)editor).getEditorInput().getTaxonNode());
300
	            }else{
301
	                if (editor.getTaxon().getTaxonNodes() != null && !editor.getTaxon().getTaxonNodes().isEmpty()){
302
	                    taxonNode = new TaxonNodeDto(editor.getTaxon().getTaxonNodes().iterator().next());
303
	                }
304
	            }
305
                if (taxonNode != null){
306
                    viewer.reveal(taxonNode);
307
                    viewer.setSelection(new StructuredSelection(taxonNode));
308
                }
309

    
310
	        }
311
	    }
312
	}
313

    
314
    public void setLinkWithTaxon(boolean linkWithTaxon) {
315
        this.linkWithTaxon = linkWithTaxon;
316
    }
317

    
318
    public boolean isLinkWithTaxon() {
319
        return linkWithTaxon;
320
    }
321

    
322
	/**
323
	 * Refresh this navigators viewer
324
	 */
325
	public void refresh() {
326
		if(getConversationHolder() != null){
327
			getConversationHolder().bind();
328
			//FIXME : Need to make sure this is a stable fix (ticket 3822)
329
			if(!getConversationHolder().isCompleted()){
330
			    getConversationHolder().commit();
331
			}
332
		}
333
		if(!viewer.getTree().isDisposed()){
334
		    viewer.refresh();
335
		}
336

    
337
		updateSyncButton();
338
	}
339

    
340
	/**
341
     * Refresh this navigators viewer
342
     */
343
    public void refresh(Set<?> objects) {
344
        for(Object obj : objects) {
345
            viewer.refresh(obj);
346
        }
347
        updateSyncButton();
348
    }
349

    
350
    /**
351
     * Refresh this navigators viewer
352
     */
353
    public void refresh(Object object) {
354
        viewer.refresh(object);
355
        updateSyncButton();
356
    }
357

    
358
	/**
359
	 * Removes all content
360
	 */
361
	public void clear() {
362
	    viewer.setInput(new EmptyRoot());
363
	}
364

    
365
	private void restore(IMemento memento, IProgressMonitor monitor) {
366
	    root = new Root(conversation);
367
		if (memento == null) {
368
		    viewer.setInput(root);
369
			return;
370
		}
371
		int mementoWork = 0;
372
		Set<TreePath> treePaths = new HashSet<TreePath>();
373
		IMemento[] treePathMementos = null;
374

    
375
		IMemento treePathsMemento = memento.getChild(TREE_PATHS);
376

    
377
		if (treePathsMemento != null) {
378
			treePathMementos = treePathsMemento.getChildren(TREE_PATH);
379
			mementoWork = treePathMementos.length;
380
		}
381
		// begin the monitor with steps for all tree paths and steps for
382
		// creating
383
		// conversation s.o., refreshing the tree and setting the paths
384
		IProgressMonitor subProgressMonitor = AbstractUtility
385
				.getSubProgressMonitor(monitor, 1);
386

    
387
		subProgressMonitor.beginTask(RESTORING_TAXON_NAVIGATOR,
388
				1 + mementoWork + 5);
389
		subProgressMonitor.subTask(RESTORING_TAXON_NAVIGATOR);
390
		subProgressMonitor.worked(1);
391

    
392
		conversation = CdmStore.createConversation();
393
		subProgressMonitor.worked(1);
394
		conversation.registerForDataStoreChanges(TaxonNavigatorE4.this);
395
		subProgressMonitor.worked(1);
396
		viewer.setInput(root);
397
		subProgressMonitor.worked(1);
398
		viewer.refresh();
399
		subProgressMonitor.worked(1);
400

    
401
		if (treePathMementos != null && treePathMementos.length > 0) {
402
			for (IMemento treePathMemento : treePathMementos) {
403
				TreePath treePath = createTreePathFromString(treePathMemento
404
						.getID());
405
				if (!subProgressMonitor.isCanceled() && treePath != null) {
406
					treePaths.add(treePath);
407
					subProgressMonitor.worked(1);
408
				}
409
			}
410
		}
411
		if (treePaths.size() > 0) {
412
			TaxonNavigatorE4.this.viewer.setExpandedTreePaths(
413
					treePaths.toArray(new TreePath[0]));
414
			subProgressMonitor.worked(1);
415
		}
416
		subProgressMonitor.done();
417
	}
418

    
419
	private TreePath createTreePathFromString(String string) {
420

    
421
		List<CdmBase> pathList = new ArrayList<CdmBase>();
422

    
423
		if (string.length() == 0) {
424
            return null;
425
        }
426

    
427
		for (String uuid : string.split(" ")) { //$NON-NLS-1$
428
			CdmBase cdmBaseObject = CdmStore.getService(
429
					ITaxonNodeService.class).find(
430
					UUID.fromString(uuid));
431
			if (cdmBaseObject == null) {
432
				// is this a tree uuid?
433
				cdmBaseObject = CdmStore.getService(
434
						IClassificationService.class).load(
435
						UUID.fromString(uuid));
436

    
437
				if (cdmBaseObject == null) {
438
                    return null;
439
                }
440
			}
441
			pathList.add(cdmBaseObject);
442
		}
443
		return new TreePath(pathList.toArray());
444
	}
445

    
446
	/**
447
	 * {@inheritDoc}
448
	 */
449
	@Override
450
	public void collapse() {
451
	    viewer.collapseAll();
452
	}
453

    
454
	/**
455
	 * {@inheritDoc}
456
	 */
457
	@Override
458
	public void expand() {
459
	    viewer.expandAll();
460
	}
461

    
462
	@Override
463
	public ConversationHolder getConversationHolder() {
464
		return conversation;
465
	}
466

    
467
	/** {@inheritDoc} */
468
	@PreDestroy
469
	public void dispose() {
470
		dataChangeBehavior = null;
471
		if (conversation != null) {
472
			conversation.unregisterForDataStoreChanges(this);
473
			conversation.close();
474
		}
475
		if(cdmEntitySession != null) {
476
		    cdmEntitySession.dispose();
477
		    cdmEntitySession = null;
478
		}
479
		if(CdmApplicationState.getCurrentDataChangeService() != null) {
480
		    CdmApplicationState.getCurrentDataChangeService().unregister(this);
481
		}
482
	}
483

    
484
	/** {@inheritDoc} */
485
	@Focus
486
	public void setFocus() {
487
		if (getConversationHolder() != null) {
488
			getConversationHolder().bind();
489
		}
490
		if(cdmEntitySession != null) {
491
		    cdmEntitySession.bind();
492
		}
493
	}
494

    
495
    public UISynchronize getSync() {
496
        return sync;
497
    }
498

    
499
    public TreeViewer getViewer() {
500
        return viewer;
501
    }
502

    
503
    public UndoContext getUndoContext() {
504
        return undoContext;
505
    }
506

    
507
	/** {@inheritDoc} */
508
	@Override
509
	public boolean postOperation(Object objectAffectedByOperation) {
510
	    viewer.refresh();
511
		return true;
512
	}
513

    
514
	@Override
515
	public boolean onComplete() {
516
		return true;
517
	}
518

    
519
	@Override
520
	public void update(Observable o, Object arg) {
521
		if(o instanceof LoginManager){
522
			refresh();
523
		}
524

    
525
	}
526
	   /** {@inheritDoc} */
527
    @Override
528
    public void update(CdmDataChangeMap changeEvents) {
529
        if (dataChangeBehavior == null) {
530
            dataChangeBehavior = new TaxonNavigatorDataChangeBehaviorE4(this);
531
        }
532

    
533
        DataChangeBridge.handleDataChange(changeEvents, dataChangeBehavior);
534
        updateSyncButton();
535

    
536
    }
537

    
538
    @Override
539
    public ICdmEntitySession getCdmEntitySession() {
540
       return cdmEntitySession;
541
    }
542

    
543
    @Override
544
    public List<TaxonNodeDto> getRootEntities() {
545
        if(root != null) {
546
            return root.getParentBeans();
547
        }
548
        return null;
549
    }
550

    
551
    @Override
552
    public void onChange(CdmChangeEvent event) {
553
        refresh();
554
        for(CdmBase cb : event.getChangedObjects()) {
555
        	if(cb instanceof TaxonNode) {
556
                TaxonNode tn = (TaxonNode)cb;
557
                if(tn.getTaxon() == null) {
558
                    viewer.refresh(tn.getClassification());
559
                } else {
560
                    viewer.refresh(cb);
561
                }
562
            } else if (cb instanceof Classification) {
563
                if ( event.getAction().equals(Action.Create)){
564
                    root.addRootNode((Classification)cb);
565
                } else if ( event.getAction().equals(Action.Delete)){
566
                    root.removeRootNode((Classification)cb);
567
                }
568
                viewer.refresh();
569
            }
570
        }
571
    }
572

    
573
    @Override
574
    public Map<Object, List<String>> getPropertyPathsMap() {
575
        Map<Object, List<String>> propertyPathsMap = new HashMap<Object, List<String>>();
576
        List<String> taxonNodePropertyPaths = Arrays.asList(new String[] {
577
                "taxon.name" //$NON-NLS-1$
578
        });
579
         propertyPathsMap.put("childNodes", taxonNodePropertyPaths); //$NON-NLS-1$
580
        return propertyPathsMap;
581
    }
582

    
583
    /**
584
     * {@inheritDoc}
585
     */
586
    @Override
587
    public void contextAboutToStop(IMemento memento, IProgressMonitor monitor) {
588
        // TODO Auto-generated method stub
589

    
590
    }
591

    
592
    /**
593
     * {@inheritDoc}
594
     */
595
    @Override
596
    public void contextStop(IMemento memento, IProgressMonitor monitor) {
597
    }
598

    
599
    /**
600
     * {@inheritDoc}
601
     */
602
    @Override
603
    public void contextStart(IMemento memento, IProgressMonitor monitor) {
604
        if(viewer!=null && viewer.getControl()!=null && !viewer.getControl().isDisposed()){
605
            init();
606
        }
607
    }
608

    
609
    /**
610
     * {@inheritDoc}
611
     */
612
    @Override
613
    public void contextRefresh(IProgressMonitor monitor) {
614
    }
615

    
616
    /**
617
     * {@inheritDoc}
618
     */
619
    @Override
620
    public void workbenchShutdown(IMemento memento, IProgressMonitor monitor) {
621
    }
622

    
623
    @Inject
624
    @Optional
625
    private void updateView(@UIEventTopic(WorkbenchEventConstants.REFRESH_NAVIGATOR)boolean refresh){
626
        if(refresh){
627
            refresh();
628
        }
629
    }
630
}
(3-3/6)