comparison dwtx/jface/text/DefaultDocumentAdapter.d @ 129:eb30df5ca28b

Added JFace Text sources
author Frank Benoit <benoit@tionex.de>
date Sat, 23 Aug 2008 19:10:48 +0200
parents
children c4fb132a086c
comparison
equal deleted inserted replaced
128:8df1d4193877 129:eb30df5ca28b
1 /*******************************************************************************
2 * Copyright (c) 2000, 2007 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.DefaultDocumentAdapter;
14
15 import dwt.dwthelper.utils;
16
17
18 import java.util.ArrayList;
19 import java.util.Iterator;
20 import java.util.List;
21
22 import dwt.DWT;
23 import dwt.custom.TextChangeListener;
24 import dwt.custom.TextChangedEvent;
25 import dwt.custom.TextChangingEvent;
26 import dwtx.core.runtime.Assert;
27
28
29 /**
30 * Default implementation of {@link dwtx.jface.text.IDocumentAdapter}.
31 * <p>
32 * <strong>Note:</strong> This adapter does not work if the widget auto-wraps the text.
33 * </p>
34 */
35 class DefaultDocumentAdapter : IDocumentAdapter, IDocumentListener, IDocumentAdapterExtension {
36
37 /** The adapted document. */
38 private IDocument fDocument;
39 /** The document clone for the non-forwarding case. */
40 private IDocument fDocumentClone;
41 /** The original content */
42 private String fOriginalContent;
43 /** The original line delimiters */
44 private String[] fOriginalLineDelimiters;
45 /** The registered text change listeners */
46 private List fTextChangeListeners= new ArrayList(1);
47 /**
48 * The remembered document event
49 * @since 2.0
50 */
51 private DocumentEvent fEvent;
52 /** The line delimiter */
53 private String fLineDelimiter= null;
54 /**
55 * Indicates whether this adapter is forwarding document changes
56 * @since 2.0
57 */
58 private bool fIsForwarding= true;
59 /**
60 * Length of document at receipt of <code>documentAboutToBeChanged</code>
61 * @since 2.1
62 */
63 private int fRememberedLengthOfDocument;
64 /**
65 * Length of first document line at receipt of <code>documentAboutToBeChanged</code>
66 * @since 2.1
67 */
68 private int fRememberedLengthOfFirstLine;
69 /**
70 * The data of the event at receipt of <code>documentAboutToBeChanged</code>
71 * @since 2.1
72 */
73 private DocumentEvent fOriginalEvent= new DocumentEvent();
74
75
76 /**
77 * Creates a new document adapter which is initially not connected to
78 * any document.
79 */
80 public DefaultDocumentAdapter() {
81 }
82
83 /**
84 * Sets the given document as the document to be adapted.
85 *
86 * @param document the document to be adapted or <code>null</code> if there is no document
87 */
88 public void setDocument(IDocument document) {
89
90 if (fDocument !is null)
91 fDocument.removePrenotifiedDocumentListener(this);
92
93 fDocument= document;
94 fLineDelimiter= null;
95
96 if (!fIsForwarding) {
97 fDocumentClone= null;
98 if (fDocument !is null) {
99 fOriginalContent= fDocument.get();
100 fOriginalLineDelimiters= fDocument.getLegalLineDelimiters();
101 } else {
102 fOriginalContent= null;
103 fOriginalLineDelimiters= null;
104 }
105 }
106
107 if (fDocument !is null)
108 fDocument.addPrenotifiedDocumentListener(this);
109 }
110
111 /*
112 * @see StyledTextContent#addTextChangeListener(TextChangeListener)
113 */
114 public void addTextChangeListener(TextChangeListener listener) {
115 Assert.isNotNull(listener);
116 if (!fTextChangeListeners.contains(listener))
117 fTextChangeListeners.add(listener);
118 }
119
120 /*
121 * @see StyledTextContent#removeTextChangeListener(TextChangeListener)
122 */
123 public void removeTextChangeListener(TextChangeListener listener) {
124 Assert.isNotNull(listener);
125 fTextChangeListeners.remove(listener);
126 }
127
128 /**
129 * Tries to repair the line information.
130 *
131 * @param document the document
132 * @see IRepairableDocument#repairLineInformation()
133 * @since 3.0
134 */
135 private void repairLineInformation(IDocument document) {
136 if (document instanceof IRepairableDocument) {
137 IRepairableDocument repairable= (IRepairableDocument) document;
138 repairable.repairLineInformation();
139 }
140 }
141
142 /**
143 * Returns the line for the given line number.
144 *
145 * @param document the document
146 * @param line the line number
147 * @return the content of the line of the given number in the given document
148 * @throws BadLocationException if the line number is invalid for the adapted document
149 * @since 3.0
150 */
151 private String doGetLine(IDocument document, int line) throws BadLocationException {
152 IRegion r= document.getLineInformation(line);
153 return document.get(r.getOffset(), r.getLength());
154 }
155
156 private IDocument getDocumentForRead() {
157 if (!fIsForwarding) {
158 if (fDocumentClone is null) {
159 String content= fOriginalContent is null ? "" : fOriginalContent; //$NON-NLS-1$
160 String[] delims= fOriginalLineDelimiters is null ? DefaultLineTracker.DELIMITERS : fOriginalLineDelimiters;
161 fDocumentClone= new DocumentClone(content, delims);
162 }
163 return fDocumentClone;
164 }
165
166 return fDocument;
167 }
168
169 /*
170 * @see StyledTextContent#getLine(int)
171 */
172 public String getLine(int line) {
173
174 IDocument document= getDocumentForRead();
175 try {
176 return doGetLine(document, line);
177 } catch (BadLocationException x) {
178 repairLineInformation(document);
179 try {
180 return doGetLine(document, line);
181 } catch (BadLocationException x2) {
182 }
183 }
184
185 DWT.error(DWT.ERROR_INVALID_ARGUMENT);
186 return null;
187 }
188
189 /*
190 * @see StyledTextContent#getLineAtOffset(int)
191 */
192 public int getLineAtOffset(int offset) {
193 IDocument document= getDocumentForRead();
194 try {
195 return document.getLineOfOffset(offset);
196 } catch (BadLocationException x) {
197 repairLineInformation(document);
198 try {
199 return document.getLineOfOffset(offset);
200 } catch (BadLocationException x2) {
201 }
202 }
203
204 DWT.error(DWT.ERROR_INVALID_ARGUMENT);
205 return -1;
206 }
207
208 /*
209 * @see StyledTextContent#getLineCount()
210 */
211 public int getLineCount() {
212 return getDocumentForRead().getNumberOfLines();
213 }
214
215 /*
216 * @see StyledTextContent#getOffsetAtLine(int)
217 */
218 public int getOffsetAtLine(int line) {
219 IDocument document= getDocumentForRead();
220 try {
221 return document.getLineOffset(line);
222 } catch (BadLocationException x) {
223 repairLineInformation(document);
224 try {
225 return document.getLineOffset(line);
226 } catch (BadLocationException x2) {
227 }
228 }
229
230 DWT.error(DWT.ERROR_INVALID_ARGUMENT);
231 return -1;
232 }
233
234 /*
235 * @see StyledTextContent#getTextRange(int, int)
236 */
237 public String getTextRange(int offset, int length) {
238 try {
239 return getDocumentForRead().get(offset, length);
240 } catch (BadLocationException x) {
241 DWT.error(DWT.ERROR_INVALID_ARGUMENT);
242 return null;
243 }
244 }
245
246 /*
247 * @see StyledTextContent#replaceTextRange(int, int, String)
248 */
249 public void replaceTextRange(int pos, int length, String text) {
250 try {
251 fDocument.replace(pos, length, text);
252 } catch (BadLocationException x) {
253 DWT.error(DWT.ERROR_INVALID_ARGUMENT);
254 }
255 }
256
257 /*
258 * @see StyledTextContent#setText(String)
259 */
260 public void setText(String text) {
261 fDocument.set(text);
262 }
263
264 /*
265 * @see StyledTextContent#getCharCount()
266 */
267 public int getCharCount() {
268 return getDocumentForRead().getLength();
269 }
270
271 /*
272 * @see StyledTextContent#getLineDelimiter()
273 */
274 public String getLineDelimiter() {
275 if (fLineDelimiter is null)
276 fLineDelimiter= TextUtilities.getDefaultLineDelimiter(fDocument);
277 return fLineDelimiter;
278 }
279
280 /*
281 * @see IDocumentListener#documentChanged(DocumentEvent)
282 */
283 public void documentChanged(DocumentEvent event) {
284 // check whether the given event is the one which was remembered
285 if (fEvent is null || event !is fEvent)
286 return;
287
288 if (isPatchedEvent(event) || (event.getOffset() is 0 && event.getLength() is fRememberedLengthOfDocument)) {
289 fLineDelimiter= null;
290 fireTextSet();
291 } else {
292 if (event.getOffset() < fRememberedLengthOfFirstLine)
293 fLineDelimiter= null;
294 fireTextChanged();
295 }
296 }
297
298 /*
299 * @see IDocumentListener#documentAboutToBeChanged(DocumentEvent)
300 */
301 public void documentAboutToBeChanged(DocumentEvent event) {
302
303 fRememberedLengthOfDocument= fDocument.getLength();
304 try {
305 fRememberedLengthOfFirstLine= fDocument.getLineLength(0);
306 } catch (BadLocationException e) {
307 fRememberedLengthOfFirstLine= -1;
308 }
309
310 fEvent= event;
311 rememberEventData(fEvent);
312 fireTextChanging();
313 }
314
315 /**
316 * Checks whether this event has been changed between <code>documentAboutToBeChanged</code> and
317 * <code>documentChanged</code>.
318 *
319 * @param event the event to be checked
320 * @return <code>true</code> if the event has been changed, <code>false</code> otherwise
321 */
322 private bool isPatchedEvent(DocumentEvent event) {
323 return fOriginalEvent.fOffset !is event.fOffset || fOriginalEvent.fLength !is event.fLength || fOriginalEvent.fText !is event.fText;
324 }
325
326 /**
327 * Makes a copy of the given event and remembers it.
328 *
329 * @param event the event to be copied
330 */
331 private void rememberEventData(DocumentEvent event) {
332 fOriginalEvent.fOffset= event.fOffset;
333 fOriginalEvent.fLength= event.fLength;
334 fOriginalEvent.fText= event.fText;
335 }
336
337 /**
338 * Sends a text changed event to all registered listeners.
339 */
340 private void fireTextChanged() {
341
342 if (!fIsForwarding)
343 return;
344
345 TextChangedEvent event= new TextChangedEvent(this);
346
347 if (fTextChangeListeners !is null && fTextChangeListeners.size() > 0) {
348 Iterator e= new ArrayList(fTextChangeListeners).iterator();
349 while (e.hasNext())
350 ((TextChangeListener) e.next()).textChanged(event);
351 }
352 }
353
354 /**
355 * Sends a text set event to all registered listeners.
356 */
357 private void fireTextSet() {
358
359 if (!fIsForwarding)
360 return;
361
362 TextChangedEvent event = new TextChangedEvent(this);
363
364 if (fTextChangeListeners !is null && fTextChangeListeners.size() > 0) {
365 Iterator e= new ArrayList(fTextChangeListeners).iterator();
366 while (e.hasNext())
367 ((TextChangeListener) e.next()).textSet(event);
368 }
369 }
370
371 /**
372 * Sends the text changing event to all registered listeners.
373 */
374 private void fireTextChanging() {
375
376 if (!fIsForwarding)
377 return;
378
379 try {
380 IDocument document= fEvent.getDocument();
381 if (document is null)
382 return;
383
384 TextChangingEvent event= new TextChangingEvent(this);
385 event.start= fEvent.fOffset;
386 event.replaceCharCount= fEvent.fLength;
387 event.replaceLineCount= document.getNumberOfLines(fEvent.fOffset, fEvent.fLength) - 1;
388 event.newText= fEvent.fText;
389 event.newCharCount= (fEvent.fText is null ? 0 : fEvent.fText.length());
390 event.newLineCount= (fEvent.fText is null ? 0 : document.computeNumberOfLines(fEvent.fText));
391
392 if (fTextChangeListeners !is null && fTextChangeListeners.size() > 0) {
393 Iterator e= new ArrayList(fTextChangeListeners).iterator();
394 while (e.hasNext())
395 ((TextChangeListener) e.next()).textChanging(event);
396 }
397
398 } catch (BadLocationException e) {
399 }
400 }
401
402 /*
403 * @see IDocumentAdapterExtension#resumeForwardingDocumentChanges()
404 * @since 2.0
405 */
406 public void resumeForwardingDocumentChanges() {
407 fIsForwarding= true;
408 fDocumentClone= null;
409 fOriginalContent= null;
410 fOriginalLineDelimiters= null;
411 fireTextSet();
412 }
413
414 /*
415 * @see IDocumentAdapterExtension#stopForwardingDocumentChanges()
416 * @since 2.0
417 */
418 public void stopForwardingDocumentChanges() {
419 fDocumentClone= null;
420 fOriginalContent= fDocument.get();
421 fOriginalLineDelimiters= fDocument.getLegalLineDelimiters();
422 fIsForwarding= false;
423 }
424 }