Skip to main content
summaryrefslogtreecommitdiffstats
blob: 1192909c708a2c8cac2222a8afa7974beb6ef2e4 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
/*******************************************************************************
 * Copyright (c) 2000, 2008 IBM Corporation and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 *
 * Contributors:
 *     IBM Corporation - initial API and implementation
 *******************************************************************************/
package org.eclipse.wst.jsdt.ui;
 
import java.io.IOException;
import java.io.Reader;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.List;

import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IConfigurationElement;
import org.eclipse.core.runtime.IExtension;
import org.eclipse.core.runtime.IExtensionPoint;
import org.eclipse.core.runtime.IExtensionRegistry;
import org.eclipse.core.runtime.Platform;
import org.eclipse.wst.jsdt.core.IBuffer;
import org.eclipse.wst.jsdt.core.IFunction;
import org.eclipse.wst.jsdt.core.IJavaScriptElement;
import org.eclipse.wst.jsdt.core.ILocalVariable;
import org.eclipse.wst.jsdt.core.IMember;
import org.eclipse.wst.jsdt.core.IOpenable;
import org.eclipse.wst.jsdt.core.ISourceRange;
import org.eclipse.wst.jsdt.core.IType;
import org.eclipse.wst.jsdt.core.ITypeHierarchy;
import org.eclipse.wst.jsdt.core.JavaScriptModelException;
import org.eclipse.wst.jsdt.core.util.SequenceReader;
import org.eclipse.wst.jsdt.internal.core.MetadataFile;
import org.eclipse.wst.jsdt.internal.corext.javadoc.JavaDocCommentReader;
import org.eclipse.wst.jsdt.internal.corext.util.MethodOverrideTester;
import org.eclipse.wst.jsdt.internal.ui.JavaScriptPlugin;
import org.eclipse.wst.jsdt.internal.ui.text.javadoc.JavaDoc2HTMLTextReader;
import org.eclipse.wst.jsdt.internal.ui.text.javadoc.OAADocReader;

/**
 * Helper needed to get the content of a Javadoc comment.
 * 
 * <p>
 * This class is not intended to be subclassed or instantiated by clients.
 * </p>
 *
 * Provisional API: This class/interface is part of an interim API that is still under development and expected to
 * change significantly before reaching stability. It is being made available at this early stage to solicit feedback
 * from pioneering adopters on the understanding that any code that uses this API will almost certainly be broken
 * (repeatedly) as the API evolves.
 */
public class JSdocContentAccess {
	
	public static final String EXTENSION_POINT= "documentationProvider"; //$NON-NLS-1$

	protected static final String TAG_DOCUMENTATIONPROVIDER = "documentationProvider"; //$NON-NLS-1$
	protected static final String ATTR_DOCUMENTATIONPROVIDER_CLASS = "class"; //$NON-NLS-1$

	private static IDocumentationReader[] docReaders;
	
	private JSdocContentAccess() {
		// do not instantiate
	}
	
	/**
	 * Gets a reader for an IMember's Javadoc comment content from the source attachment.
	 * The content does contain only the text from the comment without the Javadoc leading star characters.
	 * Returns <code>null</code> if the member does not contain a Javadoc comment or if no source is available.
	 * @param member The member to get the Javadoc of.
	 * @param allowInherited For methods with no (Javadoc) comment, the comment of the overridden class
	 * is returned if <code>allowInherited</code> is <code>true</code>.
	 * @return Returns a reader for the Javadoc comment content or <code>null</code> if the member
	 * does not contain a Javadoc comment or if no source is available
	 * @throws JavaScriptModelException is thrown when the elements javadoc can not be accessed
	 */
	public static Reader getContentReader(IMember member, boolean allowInherited) throws JavaScriptModelException {
		List readers = new ArrayList(2);
		IDocumentationReader[] docReaders = getDocReaders(member);
		for (int i = 0; i < docReaders.length; i++) {
			Reader contentReader = docReaders[i].getContentReader(member, allowInherited);
			if(contentReader != null) {
				readers.add(contentReader);
			}
		}
		
		IOpenable openable = member.getOpenable();
 		if (openable instanceof MetadataFile)
 		{
 			return new OAADocReader((MetadataFile)openable, member);
 		}
		
		IBuffer buf= openable.getBuffer();
		if (buf != null) {
			// source or attachment found
			ISourceRange javadocRange= member.getJSdocRange();
			if(javadocRange == null && member.getElementType() == IJavaScriptElement.TYPE) {
				IFunction constructor = ((IType) member).getFunction(member.getElementName(), null);
				if(constructor.exists()) {
					javadocRange = constructor.getJSdocRange();
				}
			}
			if (javadocRange != null) {
				JavaDocCommentReader reader= new JavaDocCommentReader(buf, javadocRange.getOffset(), javadocRange.getOffset() + javadocRange.getLength() - 1);
				if (!containsOnlyInheritDoc(reader, javadocRange.getLength())) {
					reader.reset();
					readers.add(reader);
				}
				else if (allowInherited && (member.getElementType() == IJavaScriptElement.METHOD)) {
					Reader hierarchyDocReader = findDocInHierarchy((IFunction) member);
					if (hierarchyDocReader != null)
						readers.add(hierarchyDocReader);
				}
			}

		}
		
		if (!readers.isEmpty()) {
			if (readers.size() == 1)
				return (Reader) readers.get(0);
			return new SequenceReader((Reader[]) readers.toArray(new Reader[readers.size()]));
		}
		return null;
	}

	/**
	 * Checks whether the given reader only returns
	 * the inheritDoc tag.
	 * 
	 * @param reader the reader
	 * @param length the length of the underlying content
	 * @return <code>true</code> if the reader only returns the inheritDoc tag
	 * 
	 */
	private static boolean containsOnlyInheritDoc(Reader reader, int length) {
		char[] content= new char[length];
		try {
			reader.read(content, 0, length);
		} catch (IOException e) {
			return false;
		}
		return new String(content).trim().equals("{@inheritDoc}"); //$NON-NLS-1$
		
	}

	/**
	 * Gets a reader for an IMember's Javadoc comment content from the source attachment.
	 * and renders the tags in HTML. 
	 * Returns <code>null</code> if the member does not contain a Javadoc comment or if no source is available.
	 * 
	 * @param member				the member to get the Javadoc of.
	 * @param allowInherited		for methods with no (Javadoc) comment, the comment of the overridden
	 * 									class is returned if <code>allowInherited</code> is <code>true</code>
	 * @param useAttachedJavadoc	if <code>true</code> Javadoc will be extracted from attached Javadoc
	 * 									if there's no source
	 * @return a reader for the Javadoc comment content in HTML or <code>null</code> if the member
	 * 			does not contain a Javadoc comment or if no source is available
	 * @throws JavaScriptModelException is thrown when the elements Javadoc can not be accessed
	 * 
	 */
	public static Reader getHTMLContentReader(IMember member, boolean allowInherited, boolean useAttachedJavadoc) throws JavaScriptModelException {
		Reader contentReader= getContentReader(member, allowInherited);
		if (contentReader != null)
		{
			IDocumentationReader[] docReaders = getDocReaders(member);
			if (docReaders.length > 0) {
				List htmlReaders = new ArrayList(docReaders.length);
				for (int i = 0; i < docReaders.length; i++) {
					Reader documentation2htmlReader = docReaders[i].getDocumentation2HTMLReader(contentReader);
					if (documentation2htmlReader != null) {
						htmlReaders.add(documentation2htmlReader);
					}
				}
				if (!htmlReaders.isEmpty()) {
					htmlReaders.add(/*0, */new JavaDoc2HTMLTextReader(contentReader));
					return new SequenceReader((Reader[]) htmlReaders.toArray(new Reader[htmlReaders.size()]));
				}
			}
			return new JavaDoc2HTMLTextReader(contentReader);
		}
		
		if (useAttachedJavadoc && member.getOpenable().getBuffer() == null) { // only if no source available
			String s= member.getAttachedJavadoc(null);
			if (s != null)
				return new StringReader(s);
		}
		return null;
	}
	
	/**
	 * Gets a reader for an ILocalDeclaration documentation comment content.
	 * and renders the tags in HTML.
	 * Returns <code>null</code> if the declaration does not contain a doc comment or if no source is available.
	 * 
	 * @param variable			the variable declaration to get the doc of.
	 * @param allowInherited		for methods with no (JSDoc) comment, the comment of the overridden
	 * 									class is returned if <code>allowInherited</code> is <code>true</code>
	 * @param useAttachedDoc	if <code>true</code> JSDoc will be extracted from attached JSDoc
	 * 									if there's no source
	 * @return a reader for the JSDoc comment content in HTML or <code>null</code> if the member
	 * 			does not contain a JSDoc comment or if no source is available
	 * @throws JavaScriptModelException is thrown when the elements JSDoc can not be accessed
	 * 
	 */
	public static Reader getHTMLContentReader(ILocalVariable variable, boolean allowInherited, boolean useAttachedDoc) throws JavaScriptModelException {
		Reader contentReader= getContentReader(variable, allowInherited);
		if (contentReader != null)
		{
			IDocumentationReader[] docReaders = new IDocumentationReader[0];//getDocReaders(declaration);
			if (docReaders.length > 0) {
				List htmlReaders = new ArrayList(docReaders.length);
				for (int i = 0; i < docReaders.length; i++) {
					Reader documentation2htmlReader = docReaders[i].getDocumentation2HTMLReader(contentReader);
					if (documentation2htmlReader != null) {
						htmlReaders.add(documentation2htmlReader);
					}
				}
				if (!htmlReaders.isEmpty()) {
					htmlReaders.add(/*0, */new JavaDoc2HTMLTextReader(contentReader));
					return new SequenceReader((Reader[]) htmlReaders.toArray(new Reader[htmlReaders.size()]));
				}
			}
			return new JavaDoc2HTMLTextReader(contentReader);
		}
		
		return null;
	}



	/**
	 * Gets a reader for an ILocalDeclaration's doc comment content from the
	 * source attachment. Returns <code>null</code> if the declaration does
	 * not have a doc comment or if no source is available.
	 * 
	 * @param declaration
	 *            The declaration to get the doc of.
	 * @param allowInherited
	 *            For methods with no doc comment, the comment of the
	 *            overridden class is returned if <code>allowInherited</code>
	 *            is <code>true</code> and this is an argument.
	 * @return Returns a reader for the doc comment content or
	 *         <code>null</code> if the declaration does not contain a doc
	 *         comment or if no source is available
	 * @throws JavaScriptModelException
	 *             is thrown when the declaration's doc can not be accessed
	 */
	public static Reader getContentReader(ILocalVariable declaration, boolean allowInherited) throws JavaScriptModelException {
		List readers = new ArrayList(2);
		IDocumentationReader[] docReaders = getDocReaders(declaration);
		for (int i = 0; i < docReaders.length; i++) {
			Reader contentReader = docReaders[i].getContentReader(declaration, allowInherited);
			if (contentReader != null) {
				readers.add(contentReader);
			}
		}

		if (!readers.isEmpty()) {
			if (readers.size() == 1)
				return (Reader) readers.get(0);
			return new SequenceReader((Reader[]) readers.toArray(new Reader[readers.size()]));
		}
		return null;
	}

	private static Reader findDocInHierarchy(IFunction method) throws JavaScriptModelException {
		IType type= method.getDeclaringType();
		if (type==null)
			return null;
		ITypeHierarchy hierarchy= type.newSupertypeHierarchy(null);
		
		MethodOverrideTester tester= new MethodOverrideTester(type, hierarchy);
		
		IType[] superTypes= hierarchy.getAllSupertypes(type);
		for (int i= 0; i < superTypes.length; i++) {
			IType curr= superTypes[i];
			IFunction overridden= tester.findOverriddenMethodInType(curr, method);
			if (overridden != null) {
				Reader reader= getContentReader(overridden, false);
				if (reader != null) {
					return reader;
				}
			}
		}
		return null;
	}		

	private static IDocumentationReader[] getDocReaders(IMember member)
	{
		if (docReaders==null)
			loadExtensions();
		List readers = new ArrayList(docReaders.length);
		for (int i = 0; i < docReaders.length; i++) {
			if (docReaders[i].appliesTo(member)) {
				readers.add(docReaders[i]);
			}
		}
		return (IDocumentationReader[]) readers.toArray(new IDocumentationReader[readers.size()]);
	}
	
	private static IDocumentationReader[] getDocReaders(ILocalVariable declaration)
	{
		if (docReaders==null)
			loadExtensions();
		List readers = new ArrayList(docReaders.length);
		for (int i = 0; i < docReaders.length; i++) {
			if (docReaders[i].appliesTo(declaration)) {
				readers.add(docReaders[i]);
			}
		}
		return (IDocumentationReader[]) readers.toArray(new IDocumentationReader[readers.size()]);
	}
	
	
	private static void loadExtensions() {
		IExtensionRegistry registry = Platform.getExtensionRegistry();
		ArrayList extList = new ArrayList();
		if (registry != null) {
			IExtensionPoint point = registry.getExtensionPoint(
					JavaScriptPlugin.getPluginId(), EXTENSION_POINT);

			if (point != null) {
				IExtension[] extensions = point.getExtensions();
				for (int i = 0; i < extensions.length; i++) {
					IConfigurationElement[] elements = extensions[i]
							.getConfigurationElements();
					for (int j = 0; j < elements.length; j++) {
						try {
							IDocumentationReader docProvider = null;
							if (elements[j].getName().equals(TAG_DOCUMENTATIONPROVIDER)) {
								  docProvider = (IDocumentationReader) elements[j]
										.createExecutableExtension(ATTR_DOCUMENTATIONPROVIDER_CLASS);
							}

							extList.add(docProvider);
						} catch (CoreException e) {
							e.printStackTrace();
						}
					}
				}
			}
		}

		 docReaders = (IDocumentationReader[]) extList
				.toArray(new IDocumentationReader[extList.size()]);
	}


	
}

Back to the top