129
|
1 /*******************************************************************************
|
|
2 * Copyright (c) 2000, 2008 IBM Corporation and others.
|
|
3 * All rights reserved. This program and the accompanying materials
|
|
4 * are made available under the terms of the Eclipse Public License v1.0
|
|
5 * which accompanies this distribution, and is available at
|
|
6 * http://www.eclipse.org/legal/epl-v10.html
|
|
7 *
|
|
8 * Contributors:
|
|
9 * IBM Corporation - initial API and implementation
|
|
10 * Port to the D programming language:
|
|
11 * Frank Benoit <benoit@tionex.de>
|
|
12 *******************************************************************************/
|
|
13 module dwtx.jface.text.source.CompositeRuler;
|
|
14
|
|
15 import dwt.dwthelper.utils;
|
|
16
|
|
17
|
|
18 import java.util.ArrayList;
|
|
19 import java.util.EventListener;
|
|
20 import java.util.HashSet;
|
|
21 import java.util.Iterator;
|
|
22 import java.util.List;
|
|
23 import java.util.Set;
|
|
24
|
|
25 import dwt.DWT;
|
|
26 import dwt.custom.StyledText;
|
|
27 import dwt.events.ControlListener;
|
|
28 import dwt.events.DisposeEvent;
|
|
29 import dwt.events.DisposeListener;
|
|
30 import dwt.events.FocusListener;
|
|
31 import dwt.events.HelpListener;
|
|
32 import dwt.events.KeyListener;
|
|
33 import dwt.events.MouseListener;
|
|
34 import dwt.events.MouseMoveListener;
|
|
35 import dwt.events.MouseTrackListener;
|
|
36 import dwt.events.PaintListener;
|
|
37 import dwt.events.TraverseListener;
|
|
38 import dwt.graphics.Font;
|
|
39 import dwt.graphics.Point;
|
|
40 import dwt.graphics.Rectangle;
|
|
41 import dwt.widgets.Canvas;
|
|
42 import dwt.widgets.Composite;
|
|
43 import dwt.widgets.Control;
|
|
44 import dwt.widgets.Display;
|
|
45 import dwt.widgets.Event;
|
|
46 import dwt.widgets.Layout;
|
|
47 import dwt.widgets.Listener;
|
|
48 import dwt.widgets.Menu;
|
|
49 import dwtx.core.runtime.Assert;
|
|
50 import dwtx.jface.text.BadLocationException;
|
|
51 import dwtx.jface.text.IDocument;
|
|
52 import dwtx.jface.text.IRegion;
|
|
53 import dwtx.jface.text.ITextViewer;
|
|
54 import dwtx.jface.text.ITextViewerExtension;
|
|
55 import dwtx.jface.text.ITextViewerExtension5;
|
|
56
|
|
57
|
|
58 /**
|
|
59 * Standard implementation of
|
|
60 * {@link dwtx.jface.text.source.IVerticalRuler}.
|
|
61 * <p>
|
|
62 * This ruler does not have a a visual representation of its own. The
|
|
63 * presentation comes from the configurable list of vertical ruler columns. Such
|
|
64 * columns must implement the
|
|
65 * {@link dwtx.jface.text.source.IVerticalRulerColumn}. interface.</p>
|
|
66 * <p>
|
|
67 * Clients may instantiate and configure this class.</p>
|
|
68 *
|
|
69 * @see dwtx.jface.text.source.IVerticalRulerColumn
|
|
70 * @see dwtx.jface.text.ITextViewer
|
|
71 * @since 2.0
|
|
72 */
|
|
73 public class CompositeRuler : IVerticalRuler, IVerticalRulerExtension, IVerticalRulerInfoExtension {
|
|
74
|
|
75
|
|
76 /**
|
|
77 * Layout of the composite vertical ruler. Arranges the list of columns.
|
|
78 */
|
|
79 class RulerLayout : Layout {
|
|
80
|
|
81 /**
|
|
82 * Creates the new ruler layout.
|
|
83 */
|
|
84 protected RulerLayout() {
|
|
85 }
|
|
86
|
|
87 /*
|
|
88 * @see Layout#computeSize(Composite, int, int, bool)
|
|
89 */
|
|
90 protected Point computeSize(Composite composite, int wHint, int hHint, bool flushCache) {
|
|
91 Control[] children= composite.getChildren();
|
|
92 Point size= new Point(0, 0);
|
|
93 for (int i= 0; i < children.length; i++) {
|
|
94 Point s= children[i].computeSize(DWT.DEFAULT, DWT.DEFAULT, flushCache);
|
|
95 size.x += s.x;
|
|
96 size.y= Math.max(size.y, s.y);
|
|
97 }
|
|
98 size.x += (Math.max(0, children.length -1) * fGap);
|
|
99 return size;
|
|
100 }
|
|
101
|
|
102 /*
|
|
103 * @see Layout#layout(Composite, bool)
|
|
104 */
|
|
105 protected void layout(Composite composite, bool flushCache) {
|
|
106 Rectangle clArea= composite.getClientArea();
|
|
107 int rulerHeight= clArea.height;
|
|
108
|
|
109 int x= 0;
|
|
110 Iterator e= fDecorators.iterator();
|
|
111 while (e.hasNext()) {
|
|
112 IVerticalRulerColumn column= (IVerticalRulerColumn) e.next();
|
|
113 int columnWidth= column.getWidth();
|
|
114 column.getControl().setBounds(x, 0, columnWidth, rulerHeight);
|
|
115 x += (columnWidth + fGap);
|
|
116 }
|
|
117 }
|
|
118 }
|
|
119
|
|
120 /**
|
|
121 * A canvas that adds listeners to all its children. Used by the implementation of the
|
|
122 * vertical ruler to propagate listener additions and removals to the ruler's columns.
|
|
123 */
|
|
124 static class CompositeRulerCanvas : Canvas {
|
|
125
|
|
126 /**
|
|
127 * Keeps the information for which event type a listener object has been added.
|
|
128 */
|
|
129 static class ListenerInfo {
|
|
130 Class fClass;
|
|
131 EventListener fListener;
|
|
132 }
|
|
133
|
|
134 /** The list of listeners added to this canvas. */
|
|
135 private List fCachedListeners= new ArrayList();
|
|
136 /**
|
|
137 * Internal listener for opening the context menu.
|
|
138 * @since 3.0
|
|
139 */
|
|
140 private Listener fMenuDetectListener;
|
|
141
|
|
142 /**
|
|
143 * Creates a new composite ruler canvas.
|
|
144 *
|
|
145 * @param parent the parent composite
|
|
146 * @param style the DWT styles
|
|
147 */
|
|
148 public CompositeRulerCanvas(Composite parent, int style) {
|
|
149 super(parent, style);
|
|
150 fMenuDetectListener= new Listener() {
|
|
151 public void handleEvent(Event event) {
|
|
152 if (event.type is DWT.MenuDetect) {
|
|
153 Menu menu= getMenu();
|
|
154 if (menu !is null) {
|
|
155 menu.setLocation(event.x, event.y);
|
|
156 menu.setVisible(true);
|
|
157 }
|
|
158 }
|
|
159 }
|
|
160 };
|
|
161 super.addDisposeListener(new DisposeListener() {
|
|
162 public void widgetDisposed(DisposeEvent e) {
|
|
163 if (fCachedListeners !is null) {
|
|
164 fCachedListeners.clear();
|
|
165 fCachedListeners= null;
|
|
166 }
|
|
167 }
|
|
168 });
|
|
169 }
|
|
170
|
|
171 /**
|
|
172 * Adds the given listener object as listener of the given type (<code>clazz</code>) to
|
|
173 * the given control.
|
|
174 *
|
|
175 * @param clazz the listener type
|
|
176 * @param control the control to add the listener to
|
|
177 * @param listener the listener to be added
|
|
178 */
|
|
179 private void addListener(Class clazz, Control control, EventListener listener) {
|
|
180 if (ControlListener.class.equals(clazz)) {
|
|
181 control. addControlListener((ControlListener) listener);
|
|
182 return;
|
|
183 }
|
|
184 if (FocusListener.class.equals(clazz)) {
|
|
185 control. addFocusListener((FocusListener) listener);
|
|
186 return;
|
|
187 }
|
|
188 if (HelpListener.class.equals(clazz)) {
|
|
189 control. addHelpListener((HelpListener) listener);
|
|
190 return;
|
|
191 }
|
|
192 if (KeyListener.class.equals(clazz)) {
|
|
193 control. addKeyListener((KeyListener) listener);
|
|
194 return;
|
|
195 }
|
|
196 if (MouseListener.class.equals(clazz)) {
|
|
197 control. addMouseListener((MouseListener) listener);
|
|
198 return;
|
|
199 }
|
|
200 if (MouseMoveListener.class.equals(clazz)) {
|
|
201 control. addMouseMoveListener((MouseMoveListener) listener);
|
|
202 return;
|
|
203 }
|
|
204 if (MouseTrackListener.class.equals(clazz)) {
|
|
205 control. addMouseTrackListener((MouseTrackListener) listener);
|
|
206 return;
|
|
207 }
|
|
208 if (PaintListener.class.equals(clazz)) {
|
|
209 control. addPaintListener((PaintListener) listener);
|
|
210 return;
|
|
211 }
|
|
212 if (TraverseListener.class.equals(clazz)) {
|
|
213 control. addTraverseListener((TraverseListener) listener);
|
|
214 return;
|
|
215 }
|
|
216 if (DisposeListener.class.equals(clazz)) {
|
|
217 control. addDisposeListener((DisposeListener) listener);
|
|
218 return;
|
|
219 }
|
|
220 }
|
|
221
|
|
222 /**
|
|
223 * Removes the given listener object as listener of the given type (<code>clazz</code>) from
|
|
224 * the given control.
|
|
225 *
|
|
226 * @param clazz the listener type
|
|
227 * @param control the control to remove the listener from
|
|
228 * @param listener the listener to be removed
|
|
229 */
|
|
230 private void removeListener(Class clazz, Control control, EventListener listener) {
|
|
231 if (ControlListener.class.equals(clazz)) {
|
|
232 control. removeControlListener((ControlListener) listener);
|
|
233 return;
|
|
234 }
|
|
235 if (FocusListener.class.equals(clazz)) {
|
|
236 control. removeFocusListener((FocusListener) listener);
|
|
237 return;
|
|
238 }
|
|
239 if (HelpListener.class.equals(clazz)) {
|
|
240 control. removeHelpListener((HelpListener) listener);
|
|
241 return;
|
|
242 }
|
|
243 if (KeyListener.class.equals(clazz)) {
|
|
244 control. removeKeyListener((KeyListener) listener);
|
|
245 return;
|
|
246 }
|
|
247 if (MouseListener.class.equals(clazz)) {
|
|
248 control. removeMouseListener((MouseListener) listener);
|
|
249 return;
|
|
250 }
|
|
251 if (MouseMoveListener.class.equals(clazz)) {
|
|
252 control. removeMouseMoveListener((MouseMoveListener) listener);
|
|
253 return;
|
|
254 }
|
|
255 if (MouseTrackListener.class.equals(clazz)) {
|
|
256 control. removeMouseTrackListener((MouseTrackListener) listener);
|
|
257 return;
|
|
258 }
|
|
259 if (PaintListener.class.equals(clazz)) {
|
|
260 control. removePaintListener((PaintListener) listener);
|
|
261 return;
|
|
262 }
|
|
263 if (TraverseListener.class.equals(clazz)) {
|
|
264 control. removeTraverseListener((TraverseListener) listener);
|
|
265 return;
|
|
266 }
|
|
267 if (DisposeListener.class.equals(clazz)) {
|
|
268 control. removeDisposeListener((DisposeListener) listener);
|
|
269 return;
|
|
270 }
|
|
271 }
|
|
272
|
|
273 /**
|
|
274 * Adds the given listener object to the internal book keeping under
|
|
275 * the given listener type (<code>clazz</code>).
|
|
276 *
|
|
277 * @param clazz the listener type
|
|
278 * @param listener the listener object
|
|
279 */
|
|
280 private void addListener(Class clazz, EventListener listener) {
|
|
281 Control[] children= getChildren();
|
|
282 for (int i= 0; i < children.length; i++) {
|
|
283 if (children[i] !is null && !children[i].isDisposed())
|
|
284 addListener(clazz, children[i], listener);
|
|
285 }
|
|
286
|
|
287 ListenerInfo info= new ListenerInfo();
|
|
288 info.fClass= clazz;
|
|
289 info.fListener= listener;
|
|
290 fCachedListeners.add(info);
|
|
291 }
|
|
292
|
|
293 /**
|
|
294 * Removes the given listener object from the internal book keeping under
|
|
295 * the given listener type (<code>clazz</code>).
|
|
296 *
|
|
297 * @param clazz the listener type
|
|
298 * @param listener the listener object
|
|
299 */
|
|
300 private void removeListener(Class clazz, EventListener listener) {
|
|
301 int length= fCachedListeners.size();
|
|
302 for (int i= 0; i < length; i++) {
|
|
303 ListenerInfo info= (ListenerInfo) fCachedListeners.get(i);
|
|
304 if (listener is info.fListener && clazz.equals(info.fClass)) {
|
|
305 fCachedListeners.remove(i);
|
|
306 break;
|
|
307 }
|
|
308 }
|
|
309
|
|
310 Control[] children= getChildren();
|
|
311 for (int i= 0; i < children.length; i++) {
|
|
312 if (children[i] !is null && !children[i].isDisposed())
|
|
313 removeListener(clazz, children[i], listener);
|
|
314 }
|
|
315 }
|
|
316
|
|
317 /**
|
|
318 * Tells this canvas that a child has been added.
|
|
319 *
|
|
320 * @param child the child
|
|
321 */
|
|
322 public void childAdded(Control child) {
|
|
323 if (child !is null && !child.isDisposed()) {
|
|
324 int length= fCachedListeners.size();
|
|
325 for (int i= 0; i < length; i++) {
|
|
326 ListenerInfo info= (ListenerInfo) fCachedListeners.get(i);
|
|
327 addListener(info.fClass, child, info.fListener);
|
|
328 }
|
|
329 child.addListener(DWT.MenuDetect, fMenuDetectListener);
|
|
330 }
|
|
331 }
|
|
332
|
|
333 /**
|
|
334 * Tells this canvas that a child has been removed.
|
|
335 *
|
|
336 * @param child the child
|
|
337 */
|
|
338 public void childRemoved(Control child) {
|
|
339 if (child !is null && !child.isDisposed()) {
|
|
340 int length= fCachedListeners.size();
|
|
341 for (int i= 0; i < length; i++) {
|
|
342 ListenerInfo info= (ListenerInfo) fCachedListeners.get(i);
|
|
343 removeListener(info.fClass, child, info.fListener);
|
|
344 }
|
|
345 child.removeListener(DWT.MenuDetect, fMenuDetectListener);
|
|
346 }
|
|
347 }
|
|
348
|
|
349 /*
|
|
350 * @see Control#removeControlListener(ControlListener)
|
|
351 */
|
|
352 public void removeControlListener(ControlListener listener) {
|
|
353 removeListener(ControlListener.class, listener);
|
|
354 super.removeControlListener(listener);
|
|
355 }
|
|
356
|
|
357 /*
|
|
358 * @see Control#removeFocusListener(FocusListener)
|
|
359 */
|
|
360 public void removeFocusListener(FocusListener listener) {
|
|
361 removeListener(FocusListener.class, listener);
|
|
362 super.removeFocusListener(listener);
|
|
363 }
|
|
364
|
|
365 /*
|
|
366 * @see Control#removeHelpListener(HelpListener)
|
|
367 */
|
|
368 public void removeHelpListener(HelpListener listener) {
|
|
369 removeListener(HelpListener.class, listener);
|
|
370 super.removeHelpListener(listener);
|
|
371 }
|
|
372
|
|
373 /*
|
|
374 * @see Control#removeKeyListener(KeyListener)
|
|
375 */
|
|
376 public void removeKeyListener(KeyListener listener) {
|
|
377 removeListener(KeyListener.class, listener);
|
|
378 super.removeKeyListener(listener);
|
|
379 }
|
|
380
|
|
381 /*
|
|
382 * @see Control#removeMouseListener(MouseListener)
|
|
383 */
|
|
384 public void removeMouseListener(MouseListener listener) {
|
|
385 removeListener(MouseListener.class, listener);
|
|
386 super.removeMouseListener(listener);
|
|
387 }
|
|
388
|
|
389 /*
|
|
390 * @see Control#removeMouseMoveListener(MouseMoveListener)
|
|
391 */
|
|
392 public void removeMouseMoveListener(MouseMoveListener listener) {
|
|
393 removeListener(MouseMoveListener.class, listener);
|
|
394 super.removeMouseMoveListener(listener);
|
|
395 }
|
|
396
|
|
397 /*
|
|
398 * @see Control#removeMouseTrackListener(MouseTrackListener)
|
|
399 */
|
|
400 public void removeMouseTrackListener(MouseTrackListener listener) {
|
|
401 removeListener(MouseTrackListener.class, listener);
|
|
402 super.removeMouseTrackListener(listener);
|
|
403 }
|
|
404
|
|
405 /*
|
|
406 * @see Control#removePaintListener(PaintListener)
|
|
407 */
|
|
408 public void removePaintListener(PaintListener listener) {
|
|
409 removeListener(PaintListener.class, listener);
|
|
410 super.removePaintListener(listener);
|
|
411 }
|
|
412
|
|
413 /*
|
|
414 * @see Control#removeTraverseListener(TraverseListener)
|
|
415 */
|
|
416 public void removeTraverseListener(TraverseListener listener) {
|
|
417 removeListener(TraverseListener.class, listener);
|
|
418 super.removeTraverseListener(listener);
|
|
419 }
|
|
420
|
|
421 /*
|
|
422 * @see Widget#removeDisposeListener(DisposeListener)
|
|
423 */
|
|
424 public void removeDisposeListener(DisposeListener listener) {
|
|
425 removeListener(DisposeListener.class, listener);
|
|
426 super.removeDisposeListener(listener);
|
|
427 }
|
|
428
|
|
429 /*
|
|
430 * @seeControl#addControlListener(ControlListener)
|
|
431 */
|
|
432 public void addControlListener(ControlListener listener) {
|
|
433 super.addControlListener(listener);
|
|
434 addListener(ControlListener.class, listener);
|
|
435 }
|
|
436
|
|
437 /*
|
|
438 * @see Control#addFocusListener(FocusListener)
|
|
439 */
|
|
440 public void addFocusListener(FocusListener listener) {
|
|
441 super.addFocusListener(listener);
|
|
442 addListener(FocusListener.class, listener);
|
|
443 }
|
|
444
|
|
445 /*
|
|
446 * @see Control#addHelpListener(HelpListener)
|
|
447 */
|
|
448 public void addHelpListener(HelpListener listener) {
|
|
449 super.addHelpListener(listener);
|
|
450 addListener(HelpListener.class, listener);
|
|
451 }
|
|
452
|
|
453 /*
|
|
454 * @see Control#addKeyListener(KeyListener)
|
|
455 */
|
|
456 public void addKeyListener(KeyListener listener) {
|
|
457 super.addKeyListener(listener);
|
|
458 addListener(KeyListener.class, listener);
|
|
459 }
|
|
460
|
|
461 /*
|
|
462 * @see Control#addMouseListener(MouseListener)
|
|
463 */
|
|
464 public void addMouseListener(MouseListener listener) {
|
|
465 super.addMouseListener(listener);
|
|
466 addListener(MouseListener.class, listener);
|
|
467 }
|
|
468
|
|
469 /*
|
|
470 * @see Control#addMouseMoveListener(MouseMoveListener)
|
|
471 */
|
|
472 public void addMouseMoveListener(MouseMoveListener listener) {
|
|
473 super.addMouseMoveListener(listener);
|
|
474 addListener(MouseMoveListener.class, listener);
|
|
475 }
|
|
476
|
|
477 /*
|
|
478 * @see Control#addMouseTrackListener(MouseTrackListener)
|
|
479 */
|
|
480 public void addMouseTrackListener(MouseTrackListener listener) {
|
|
481 super.addMouseTrackListener(listener);
|
|
482 addListener(MouseTrackListener.class, listener);
|
|
483 }
|
|
484
|
|
485 /*
|
|
486 * @seeControl#addPaintListener(PaintListener)
|
|
487 */
|
|
488 public void addPaintListener(PaintListener listener) {
|
|
489 super.addPaintListener(listener);
|
|
490 addListener(PaintListener.class, listener);
|
|
491 }
|
|
492
|
|
493 /*
|
|
494 * @see Control#addTraverseListener(TraverseListener)
|
|
495 */
|
|
496 public void addTraverseListener(TraverseListener listener) {
|
|
497 super.addTraverseListener(listener);
|
|
498 addListener(TraverseListener.class, listener);
|
|
499 }
|
|
500
|
|
501 /*
|
|
502 * @see Widget#addDisposeListener(DisposeListener)
|
|
503 */
|
|
504 public void addDisposeListener(DisposeListener listener) {
|
|
505 super.addDisposeListener(listener);
|
|
506 addListener(DisposeListener.class, listener);
|
|
507 }
|
|
508 }
|
|
509
|
|
510 /** The ruler's viewer */
|
|
511 private ITextViewer fTextViewer;
|
|
512 /** The ruler's canvas to which to add the ruler columns */
|
|
513 private CompositeRulerCanvas fComposite;
|
|
514 /** The ruler's annotation model */
|
|
515 private IAnnotationModel fModel;
|
|
516 /** The list of columns */
|
|
517 private List fDecorators= new ArrayList(2);
|
|
518 /** The cached location of the last mouse button activity */
|
|
519 private Point fLocation= new Point(-1, -1);
|
|
520 /** The cached line of the list mouse button activity */
|
|
521 private int fLastMouseButtonActivityLine= -1;
|
|
522 /** The gap between the individual columns of this composite ruler */
|
|
523 private int fGap;
|
|
524 /**
|
|
525 * The set of annotation listeners.
|
|
526 * @since 3.0
|
|
527 */
|
|
528 private Set fAnnotationListeners= new HashSet();
|
|
529
|
|
530
|
|
531 /**
|
|
532 * Constructs a new composite vertical ruler.
|
|
533 */
|
|
534 public CompositeRuler() {
|
|
535 this(0);
|
|
536 }
|
|
537
|
|
538 /**
|
|
539 * Constructs a new composite ruler with the given gap between its columns.
|
|
540 *
|
|
541 * @param gap
|
|
542 */
|
|
543 public CompositeRuler(int gap) {
|
|
544 fGap= gap;
|
|
545 }
|
|
546
|
|
547 /**
|
|
548 * Inserts the given column at the specified slot to this composite ruler.
|
|
549 * Columns are counted from left to right.
|
|
550 *
|
|
551 * @param index the index
|
|
552 * @param rulerColumn the decorator to be inserted
|
|
553 */
|
|
554 public void addDecorator(int index, IVerticalRulerColumn rulerColumn) {
|
|
555 rulerColumn.setModel(getModel());
|
|
556
|
|
557 if (index > fDecorators.size())
|
|
558 fDecorators.add(rulerColumn);
|
|
559 else
|
|
560 fDecorators.add(index, rulerColumn);
|
|
561
|
|
562 if (fComposite !is null && !fComposite.isDisposed()) {
|
|
563 rulerColumn.createControl(this, fComposite);
|
|
564 fComposite.childAdded(rulerColumn.getControl());
|
|
565 layoutTextViewer();
|
|
566 }
|
|
567 }
|
|
568
|
|
569 /**
|
|
570 * Removes the decorator in the specified slot from this composite ruler.
|
|
571 *
|
|
572 * @param index the index
|
|
573 */
|
|
574 public void removeDecorator(int index) {
|
|
575 IVerticalRulerColumn rulerColumn= (IVerticalRulerColumn) fDecorators.get(index);
|
|
576 removeDecorator(rulerColumn);
|
|
577 }
|
|
578
|
|
579 /**
|
|
580 * Removes the given decorator from the composite ruler.
|
|
581 *
|
|
582 * @param rulerColumn the ruler column to be removed
|
|
583 * @since 3.0
|
|
584 */
|
|
585 public void removeDecorator(IVerticalRulerColumn rulerColumn) {
|
|
586 fDecorators.remove(rulerColumn);
|
|
587 if (rulerColumn !is null) {
|
|
588 Control cc= rulerColumn.getControl();
|
|
589 if (cc !is null && !cc.isDisposed()) {
|
|
590 fComposite.childRemoved(cc);
|
|
591 cc.dispose();
|
|
592 }
|
|
593 }
|
|
594 layoutTextViewer();
|
|
595 }
|
|
596
|
|
597 /**
|
|
598 * Layouts the text viewer. This also causes this ruler to get
|
|
599 * be layouted.
|
|
600 */
|
|
601 private void layoutTextViewer() {
|
|
602
|
|
603 Control parent= fTextViewer.getTextWidget();
|
|
604
|
|
605 if (fTextViewer instanceof ITextViewerExtension) {
|
|
606 ITextViewerExtension extension= (ITextViewerExtension) fTextViewer;
|
|
607 parent= extension.getControl();
|
|
608 }
|
|
609
|
|
610 if (parent instanceof Composite && !parent.isDisposed())
|
|
611 ((Composite) parent).layout(true);
|
|
612 }
|
|
613
|
|
614 /*
|
|
615 * @see IVerticalRuler#getControl()
|
|
616 */
|
|
617 public Control getControl() {
|
|
618 return fComposite;
|
|
619 }
|
|
620
|
|
621 /*
|
|
622 * @see IVerticalRuler#createControl(Composite, ITextViewer)
|
|
623 */
|
|
624 public Control createControl(Composite parent, ITextViewer textViewer) {
|
|
625
|
|
626 fTextViewer= textViewer;
|
|
627
|
|
628 fComposite= new CompositeRulerCanvas(parent, DWT.NONE);
|
|
629 fComposite.setLayout(new RulerLayout());
|
|
630
|
|
631 Iterator iter= fDecorators.iterator();
|
|
632 while (iter.hasNext()) {
|
|
633 IVerticalRulerColumn column= (IVerticalRulerColumn) iter.next();
|
|
634 column.createControl(this, fComposite);
|
|
635 fComposite.childAdded(column.getControl());
|
|
636 }
|
|
637
|
|
638 return fComposite;
|
|
639 }
|
|
640
|
|
641 /*
|
|
642 * @see IVerticalRuler#setModel(IAnnotationModel)
|
|
643 */
|
|
644 public void setModel(IAnnotationModel model) {
|
|
645
|
|
646 fModel= model;
|
|
647
|
|
648 Iterator e= fDecorators.iterator();
|
|
649 while (e.hasNext()) {
|
|
650 IVerticalRulerColumn column= (IVerticalRulerColumn) e.next();
|
|
651 column.setModel(model);
|
|
652 }
|
|
653 }
|
|
654
|
|
655 /*
|
|
656 * @see IVerticalRuler#getModel()
|
|
657 */
|
|
658 public IAnnotationModel getModel() {
|
|
659 return fModel;
|
|
660 }
|
|
661
|
|
662 /*
|
|
663 * @see IVerticalRuler#update()
|
|
664 */
|
|
665 public void update() {
|
|
666 if (fComposite !is null && !fComposite.isDisposed()) {
|
|
667 Display d= fComposite.getDisplay();
|
|
668 if (d !is null) {
|
|
669 d.asyncExec(new Runnable() {
|
|
670 public void run() {
|
|
671 immediateUpdate();
|
|
672 }
|
|
673 });
|
|
674 }
|
|
675 }
|
|
676 }
|
|
677
|
|
678 /**
|
|
679 * Immediately redraws the entire ruler (without asynchronous posting).
|
|
680 *
|
|
681 * @since 3.2
|
|
682 */
|
|
683 public void immediateUpdate() {
|
|
684 Iterator e= fDecorators.iterator();
|
|
685 while (e.hasNext()) {
|
|
686 IVerticalRulerColumn column= (IVerticalRulerColumn) e.next();
|
|
687 column.redraw();
|
|
688 }
|
|
689 }
|
|
690
|
|
691 /*
|
|
692 * @see IVerticalRulerExtension#setFont(Font)
|
|
693 */
|
|
694 public void setFont(Font font) {
|
|
695 Iterator e= fDecorators.iterator();
|
|
696 while (e.hasNext()) {
|
|
697 IVerticalRulerColumn column= (IVerticalRulerColumn) e.next();
|
|
698 column.setFont(font);
|
|
699 }
|
|
700 }
|
|
701
|
|
702 /*
|
|
703 * @see IVerticalRulerInfo#getWidth()
|
|
704 */
|
|
705 public int getWidth() {
|
|
706 int width= 0;
|
|
707 Iterator e= fDecorators.iterator();
|
|
708 while (e.hasNext()) {
|
|
709 IVerticalRulerColumn column= (IVerticalRulerColumn) e.next();
|
|
710 width += (column.getWidth() + fGap);
|
|
711 }
|
|
712 return Math.max(0, width - fGap);
|
|
713 }
|
|
714
|
|
715 /*
|
|
716 * @see IVerticalRulerInfo#getLineOfLastMouseButtonActivity()
|
|
717 */
|
|
718 public int getLineOfLastMouseButtonActivity() {
|
|
719 if (fLastMouseButtonActivityLine is -1)
|
|
720 fLastMouseButtonActivityLine= toDocumentLineNumber(fLocation.y);
|
|
721 else if (fTextViewer.getDocument() is null || fLastMouseButtonActivityLine >= fTextViewer.getDocument().getNumberOfLines())
|
|
722 fLastMouseButtonActivityLine= -1;
|
|
723 return fLastMouseButtonActivityLine;
|
|
724 }
|
|
725
|
|
726 /*
|
|
727 * @see IVerticalRulerInfo#toDocumentLineNumber(int)
|
|
728 */
|
|
729 public int toDocumentLineNumber(int y_coordinate) {
|
|
730 if (fTextViewer is null || y_coordinate is -1)
|
|
731 return -1;
|
|
732
|
|
733 StyledText text= fTextViewer.getTextWidget();
|
|
734 int line= text.getLineIndex(y_coordinate);
|
|
735
|
|
736 if (line is text.getLineCount() - 1) {
|
|
737 // check whether y_coordinate exceeds last line
|
|
738 if (y_coordinate > text.getLinePixel(line + 1))
|
|
739 return -1;
|
|
740 }
|
|
741
|
|
742 return widgetLine2ModelLine(fTextViewer, line);
|
|
743 }
|
|
744
|
|
745 /**
|
|
746 * Returns the line in the given viewer's document that correspond to the given
|
|
747 * line of the viewer's widget.
|
|
748 *
|
|
749 * @param viewer the viewer
|
|
750 * @param widgetLine the widget line
|
|
751 * @return the corresponding line the viewer's document
|
|
752 * @since 2.1
|
|
753 */
|
|
754 protected final static int widgetLine2ModelLine(ITextViewer viewer, int widgetLine) {
|
|
755
|
|
756 if (viewer instanceof ITextViewerExtension5) {
|
|
757 ITextViewerExtension5 extension= (ITextViewerExtension5) viewer;
|
|
758 return extension.widgetLine2ModelLine(widgetLine);
|
|
759 }
|
|
760
|
|
761 try {
|
|
762 IRegion r= viewer.getVisibleRegion();
|
|
763 IDocument d= viewer.getDocument();
|
|
764 return widgetLine += d.getLineOfOffset(r.getOffset());
|
|
765 } catch (BadLocationException x) {
|
|
766 }
|
|
767 return widgetLine;
|
|
768 }
|
|
769
|
|
770 /**
|
|
771 * Returns this ruler's text viewer.
|
|
772 *
|
|
773 * @return this ruler's text viewer
|
|
774 */
|
|
775 public ITextViewer getTextViewer() {
|
|
776 return fTextViewer;
|
|
777 }
|
|
778
|
|
779 /*
|
|
780 * @see IVerticalRulerExtension#setLocationOfLastMouseButtonActivity(int, int)
|
|
781 */
|
|
782 public void setLocationOfLastMouseButtonActivity(int x, int y) {
|
|
783 fLocation.x= x;
|
|
784 fLocation.y= y;
|
|
785 fLastMouseButtonActivityLine= -1;
|
|
786 }
|
|
787
|
|
788 /**
|
|
789 * Returns an iterator over the <code>IVerticalRulerColumns</code> that make up this
|
|
790 * composite column.
|
|
791 *
|
|
792 * @return an iterator over the contained columns.
|
|
793 * @since 3.0
|
|
794 */
|
|
795 public Iterator getDecoratorIterator() {
|
|
796 Assert.isNotNull(fDecorators, "fDecorators must be initialized"); //$NON-NLS-1$
|
|
797 return fDecorators.iterator();
|
|
798 }
|
|
799
|
|
800 /*
|
|
801 * @see dwtx.jface.text.source.IVerticalRulerInfoExtension#getHover()
|
|
802 * @since 3.0
|
|
803 */
|
|
804 public IAnnotationHover getHover() {
|
|
805 return null;
|
|
806 }
|
|
807
|
|
808 /*
|
|
809 * @see dwtx.jface.text.source.IVerticalRulerInfoExtension#addVerticalRulerListener(dwtx.jface.text.source.IVerticalRulerListener)
|
|
810 * @since 3.0
|
|
811 */
|
|
812 public void addVerticalRulerListener(IVerticalRulerListener listener) {
|
|
813 fAnnotationListeners.add(listener);
|
|
814 }
|
|
815
|
|
816 /*
|
|
817 * @see dwtx.jface.text.source.IVerticalRulerInfoExtension#removeVerticalRulerListener(dwtx.jface.text.source.IVerticalRulerListener)
|
|
818 * @since 3.0
|
|
819 */
|
|
820 public void removeVerticalRulerListener(IVerticalRulerListener listener) {
|
|
821 fAnnotationListeners.remove(listener);
|
|
822 }
|
|
823
|
|
824 /**
|
|
825 * Fires the annotation selected event to all registered vertical ruler
|
|
826 * listeners.
|
|
827 * TODO use robust iterators
|
|
828 *
|
|
829 * @param event the event to fire
|
|
830 * @since 3.0
|
|
831 */
|
|
832 public void fireAnnotationSelected(VerticalRulerEvent event) {
|
|
833 // forward to listeners
|
|
834 for (Iterator it= fAnnotationListeners.iterator(); it.hasNext();) {
|
|
835 IVerticalRulerListener listener= (IVerticalRulerListener) it.next();
|
|
836 listener.annotationSelected(event);
|
|
837 }
|
|
838 }
|
|
839
|
|
840 /**
|
|
841 * Fires the annotation default selected event to all registered vertical
|
|
842 * ruler listeners.
|
|
843 * TODO use robust iterators
|
|
844 *
|
|
845 * @param event the event to fire
|
|
846 * @since 3.0
|
|
847 */
|
|
848 public void fireAnnotationDefaultSelected(VerticalRulerEvent event) {
|
|
849 // forward to listeners
|
|
850 for (Iterator it= fAnnotationListeners.iterator(); it.hasNext();) {
|
|
851 IVerticalRulerListener listener= (IVerticalRulerListener) it.next();
|
|
852 listener.annotationDefaultSelected(event);
|
|
853 }
|
|
854 }
|
|
855
|
|
856 /**
|
|
857 * Informs all registered vertical ruler listeners that the content menu on a selected annotation\
|
|
858 * is about to be shown.
|
|
859 * TODO use robust iterators
|
|
860 *
|
|
861 * @param event the event to fire
|
|
862 * @param menu the menu that is about to be shown
|
|
863 * @since 3.0
|
|
864 */
|
|
865 public void fireAnnotationContextMenuAboutToShow(VerticalRulerEvent event, Menu menu) {
|
|
866 // forward to listeners
|
|
867 for (Iterator it= fAnnotationListeners.iterator(); it.hasNext();) {
|
|
868 IVerticalRulerListener listener= (IVerticalRulerListener) it.next();
|
|
869 listener.annotationContextMenuAboutToShow(event, menu);
|
|
870 }
|
|
871 }
|
|
872
|
|
873 /**
|
|
874 * Relayouts the receiver.
|
|
875 *
|
|
876 * @since 3.3
|
|
877 */
|
|
878 public void relayout() {
|
|
879 layoutTextViewer();
|
|
880 }
|
|
881 }
|