446 lines
17 KiB
Java
446 lines
17 KiB
Java
|
/*
|
||
|
* Copyright 1998-2001 Sun Microsystems, Inc. All Rights Reserved.
|
||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||
|
*
|
||
|
* This code is free software; you can redistribute it and/or modify it
|
||
|
* under the terms of the GNU General Public License version 2 only, as
|
||
|
* published by the Free Software Foundation. Sun designates this
|
||
|
* particular file as subject to the "Classpath" exception as provided
|
||
|
* by Sun in the LICENSE file that accompanied this code.
|
||
|
*
|
||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||
|
* accompanied this code).
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License version
|
||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||
|
*
|
||
|
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||
|
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||
|
* have any questions.
|
||
|
*/
|
||
|
|
||
|
import com.sun.javadoc.ClassDoc;
|
||
|
import com.sun.javadoc.MethodDoc;
|
||
|
import com.sun.javadoc.RootDoc;
|
||
|
import com.sun.javadoc.Tag;
|
||
|
|
||
|
import java.beans.Introspector;
|
||
|
|
||
|
import java.util.Enumeration;
|
||
|
import java.util.Hashtable;
|
||
|
import java.util.HashMap;
|
||
|
import java.util.StringTokenizer;
|
||
|
|
||
|
/**
|
||
|
* Properties supported and tag syntax:
|
||
|
*
|
||
|
* @beaninfo
|
||
|
* bound: flag
|
||
|
* constrained: flag
|
||
|
* expert: flag
|
||
|
* hidden: flag
|
||
|
* preferred: flag
|
||
|
* description: string
|
||
|
* displayname: string
|
||
|
* propertyeditorclass: string (with dots: foo.bar.MyPropertyEditor
|
||
|
* customizerclass: string (w/dots: foo.bar.MyCustomizer)
|
||
|
* attribute: key1 value1
|
||
|
* attribute: key2 value2
|
||
|
*
|
||
|
* TODO: getValue and genDocletInfo needs some cleaning.
|
||
|
*
|
||
|
* @author Hans Muller
|
||
|
* @author Rich Schiavi
|
||
|
* @author Mark Davidson
|
||
|
*/
|
||
|
public class GenDocletBeanInfo {
|
||
|
|
||
|
static String[] ATTRIBUTE_NAMES = { "bound",
|
||
|
"constrained",
|
||
|
"expert",
|
||
|
"hidden",
|
||
|
"preferred",
|
||
|
"displayname",
|
||
|
"propertyeditorclass",
|
||
|
"customizerclass",
|
||
|
"displayname",
|
||
|
"description",
|
||
|
"enum",
|
||
|
"attribute" };
|
||
|
private static boolean DEBUG = false;
|
||
|
|
||
|
private static String fileDir = "";
|
||
|
private static String templateDir = "";
|
||
|
|
||
|
public static final String TRUE = "true";
|
||
|
public static final String FALSE = "false";
|
||
|
|
||
|
/**
|
||
|
* Method called from the javadoc environment to determint the options length.
|
||
|
* Doclet options:
|
||
|
* -t template location
|
||
|
* -d outputdir
|
||
|
* -x true Enable debug output.
|
||
|
*/
|
||
|
public static int optionLength(String option) {
|
||
|
// remind: this needs to be cleaned up
|
||
|
if (option.equals("-t"))
|
||
|
return 2;
|
||
|
if (option.equals("-d"))
|
||
|
return 2;
|
||
|
if (option.equals("-x"))
|
||
|
return 2;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/** @beaninfo
|
||
|
* bound:true
|
||
|
* constrained:false
|
||
|
* expert:true
|
||
|
* hidden:true
|
||
|
* preferred:false
|
||
|
* description: the description of this method can
|
||
|
* do all sorts of funky things. if it \n
|
||
|
* is indented like this, we have to remove
|
||
|
* all char spaces greater than 2 and also any hard-coded \n
|
||
|
* newline characters and all newlines
|
||
|
* displayname: theString
|
||
|
* propertyeditorclass: foo.bar.MyPropertyEditorClass
|
||
|
* customizerclass: foo.bar.MyCustomizerClass
|
||
|
* attribute:key1 value1
|
||
|
* attribute: key2 value2
|
||
|
*
|
||
|
*/
|
||
|
public static boolean start(RootDoc doc) {
|
||
|
readOptions(doc.options());
|
||
|
|
||
|
if (templateDir.length() == 0) {
|
||
|
System.err.println("-t option not specified");
|
||
|
return false;
|
||
|
}
|
||
|
if (fileDir.length() == 0) {
|
||
|
System.err.println("-d option not specified");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
GenSwingBeanInfo generator = new GenSwingBeanInfo(fileDir, templateDir, DEBUG);
|
||
|
Hashtable dochash = new Hashtable();
|
||
|
DocBeanInfo dbi;
|
||
|
|
||
|
/* "javadoc Foo.java Bar.java" will return:
|
||
|
* "Foo Foo.I1 Foo.I2 Bar Bar.I1 Bar.I2"
|
||
|
* i.e., with all the innerclasses of classes specified in the command
|
||
|
* line. We don't want to generate BeanInfo for any of these inner
|
||
|
* classes, so we ignore these by remembering what the last outer
|
||
|
* class was. A hack, I admit, but makes the build faster.
|
||
|
*/
|
||
|
String previousClass = null;
|
||
|
|
||
|
ClassDoc[] classes = doc.classes();
|
||
|
|
||
|
for (int cnt = 0; cnt < classes.length; cnt++) {
|
||
|
String className = classes[cnt].qualifiedName();
|
||
|
if (previousClass != null &&
|
||
|
className.startsWith(previousClass) &&
|
||
|
className.charAt(previousClass.length()) == '.') {
|
||
|
continue;
|
||
|
}
|
||
|
previousClass = className;
|
||
|
|
||
|
// XXX - debug
|
||
|
System.out.println("\n>>> Generating beaninfo for " + className + "...");
|
||
|
|
||
|
// Examine the javadoc tags and look for the the @beaninfo tag
|
||
|
// This first block looks at the javadoc for the class
|
||
|
Tag[] tags = classes[cnt].tags();
|
||
|
for (int i = 0; i < tags.length; i++) {
|
||
|
if (tags[i].kind().equalsIgnoreCase("@beaninfo")) {
|
||
|
if (DEBUG)
|
||
|
System.out.println("GenDocletBeanInfo: found @beaninfo tagged Class: " + tags[i].text());
|
||
|
dbi = genDocletInfo(tags[i].text(), classes[cnt].name());
|
||
|
dochash.put(dbi.name, dbi);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// This block looks at the javadoc for the class methods.
|
||
|
int startPos = -1;
|
||
|
MethodDoc[] methods = classes[cnt].methods();
|
||
|
for (int j = 0; j < methods.length; j++) {
|
||
|
// actually don't "introspect" - look for all
|
||
|
// methods with a @beaninfo tag
|
||
|
tags = methods[j].tags();
|
||
|
for (int x = 0; x < tags.length; x++){
|
||
|
if (tags[x].kind().equalsIgnoreCase("@beaninfo")){
|
||
|
if ((methods[j].name().startsWith("get")) ||
|
||
|
(methods[j].name().startsWith("set")))
|
||
|
startPos = 3;
|
||
|
else if (methods[j].name().startsWith("is"))
|
||
|
startPos = 2;
|
||
|
else
|
||
|
startPos = 0;
|
||
|
String propDesc =
|
||
|
Introspector.decapitalize((methods[j].name()).substring(startPos));
|
||
|
if (DEBUG)
|
||
|
System.out.println("GenDocletBeanInfo: found @beaninfo tagged Method: " + tags[x].text());
|
||
|
dbi = genDocletInfo(tags[x].text(), propDesc);
|
||
|
dochash.put(dbi.name, dbi);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (DEBUG) {
|
||
|
// dump our classes doc beaninfo
|
||
|
System.out.println(">>>>DocletBeanInfo for class: " + classes[cnt].name());
|
||
|
Enumeration e = dochash.elements();
|
||
|
while (e.hasMoreElements()) {
|
||
|
DocBeanInfo db = (DocBeanInfo)e.nextElement();
|
||
|
System.out.println(db.toString());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Use the generator to create the beaninfo code for the class.
|
||
|
generator.genBeanInfo(classes[cnt].containingPackage().name(),
|
||
|
classes[cnt].name(), dochash);
|
||
|
// reset the values!
|
||
|
dochash.clear();
|
||
|
} // end for loop
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Reads the command line options.
|
||
|
* Side Effect, sets class variables templateDir, fileDir and DEBUG
|
||
|
*/
|
||
|
private static void readOptions(String[][] options) {
|
||
|
// Parse the command line args
|
||
|
for (int i = 0; i < options.length; i++){
|
||
|
if (options[i][0].equals("-t")) {
|
||
|
templateDir = options[i][1];
|
||
|
} else if (options[i][0].equals("-d")) {
|
||
|
fileDir = options[i][1];
|
||
|
} else if (options[i][0].equals("-x")){
|
||
|
if (options[i][1].equals("true"))
|
||
|
DEBUG=true;
|
||
|
else
|
||
|
DEBUG=false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create a "BeanInfo" data structure from the tag. This is a data structure
|
||
|
* which contains all beaninfo data for a method or a class.
|
||
|
*
|
||
|
* @param text All the text after the @beaninfo tag.
|
||
|
* @param name Name of the property i.e., mnemonic for setMnemonic
|
||
|
*/
|
||
|
private static DocBeanInfo genDocletInfo(String text, String name) {
|
||
|
int beanflags = 0;
|
||
|
String desc = "null";
|
||
|
String displayname = "null";
|
||
|
String propertyeditorclass = "null";
|
||
|
String customizerclass = "null";
|
||
|
String value = "null";
|
||
|
HashMap attribs = null;
|
||
|
HashMap enums = null;
|
||
|
|
||
|
int index;
|
||
|
|
||
|
for (int j = 0; j < ATTRIBUTE_NAMES.length; j++){
|
||
|
index = 0;
|
||
|
if ((index = text.indexOf(ATTRIBUTE_NAMES[j])) != -1){
|
||
|
value = getValue((text).substring(index),ATTRIBUTE_NAMES[j]);
|
||
|
|
||
|
if (ATTRIBUTE_NAMES[j].equalsIgnoreCase("attribute")) {
|
||
|
attribs = getAttributeMap(value, " ");
|
||
|
}
|
||
|
if (ATTRIBUTE_NAMES[j].equalsIgnoreCase("enum")) {
|
||
|
enums = getAttributeMap(value, " \n");
|
||
|
}
|
||
|
else if (ATTRIBUTE_NAMES[j].equals("displayname")){
|
||
|
displayname = value;
|
||
|
}
|
||
|
else if (ATTRIBUTE_NAMES[j].equalsIgnoreCase("propertyeditorclass")) {
|
||
|
propertyeditorclass = value;
|
||
|
}
|
||
|
else if (ATTRIBUTE_NAMES[j].equalsIgnoreCase("customizerclass")){
|
||
|
customizerclass = value;
|
||
|
}
|
||
|
else if ((ATTRIBUTE_NAMES[j].equalsIgnoreCase("bound"))
|
||
|
&& (value.equalsIgnoreCase(TRUE)))
|
||
|
beanflags = beanflags | DocBeanInfo.BOUND;
|
||
|
else if ((ATTRIBUTE_NAMES[j].equalsIgnoreCase("expert"))
|
||
|
&& (value.equalsIgnoreCase(TRUE)))
|
||
|
beanflags = beanflags | DocBeanInfo.EXPERT;
|
||
|
else if ((ATTRIBUTE_NAMES[j].equalsIgnoreCase("constrained"))
|
||
|
&& (value.equalsIgnoreCase(TRUE)))
|
||
|
beanflags = beanflags | DocBeanInfo.CONSTRAINED;
|
||
|
else if ((ATTRIBUTE_NAMES[j].equalsIgnoreCase("hidden"))
|
||
|
&& (value.equalsIgnoreCase(TRUE)))
|
||
|
beanflags = beanflags | DocBeanInfo.HIDDEN;
|
||
|
else if ((ATTRIBUTE_NAMES[j].equalsIgnoreCase("preferred"))
|
||
|
&& (value.equalsIgnoreCase(TRUE)))
|
||
|
beanflags = beanflags | DocBeanInfo.PREFERRED;
|
||
|
else if (ATTRIBUTE_NAMES[j].equalsIgnoreCase("description")){
|
||
|
desc = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
/** here we create our doclet-beaninfo data structure, which we read in
|
||
|
* later if it has anything worthwhile
|
||
|
*/
|
||
|
|
||
|
// Construct a new Descriptor class
|
||
|
return new DocBeanInfo(name, beanflags, desc,displayname,
|
||
|
propertyeditorclass, customizerclass,
|
||
|
attribs, enums);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses the substring and returns the cleaned up value for the attribute.
|
||
|
* @param substring Full String of the attrib tag.
|
||
|
* i.e., "attribute: visualUpdate true" will return "visualUpdate true";
|
||
|
*/
|
||
|
private static String getValue(String substring, String prop) {
|
||
|
StringTokenizer t;
|
||
|
String value = "null";
|
||
|
|
||
|
try {
|
||
|
/** if the ATTRIBUTE_NAMES is NOT the description, then we
|
||
|
* parse until newline
|
||
|
* if it is the description we read until the next token
|
||
|
* and then look for a match in the last MAXMATCH index
|
||
|
* and truncate the description
|
||
|
* if it is the attribute we wead until no more
|
||
|
*/
|
||
|
if (prop.equalsIgnoreCase("attribute")){
|
||
|
StringBuffer tmp = new StringBuffer();
|
||
|
try {
|
||
|
t = new StringTokenizer(substring, " :\n");
|
||
|
t.nextToken().trim();//the prop
|
||
|
// we want to return : key1 value1 key2 value2
|
||
|
while (t.hasMoreTokens()){
|
||
|
tmp.append(t.nextToken().trim()).append(" ");
|
||
|
tmp.append(t.nextToken().trim()).append(" ");
|
||
|
String test = t.nextToken().trim();
|
||
|
if (!(test.equalsIgnoreCase("attribute")))
|
||
|
break;
|
||
|
}
|
||
|
} catch (Exception e){
|
||
|
}
|
||
|
value = tmp.toString();
|
||
|
}
|
||
|
else if (prop.equalsIgnoreCase("enum")){
|
||
|
t = new StringTokenizer(substring, ":");
|
||
|
t.nextToken().trim(); // the prop we already know
|
||
|
StringBuffer tmp = new StringBuffer(t.nextToken().trim());
|
||
|
for (int i = 0; i < ATTRIBUTE_NAMES.length; i++){
|
||
|
if (tmp.toString().endsWith(ATTRIBUTE_NAMES[i])){
|
||
|
int len = ATTRIBUTE_NAMES[i].length();
|
||
|
// trim off that
|
||
|
tmp.setLength(tmp.length() - len);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
value = tmp.toString();
|
||
|
}
|
||
|
else if (prop.equalsIgnoreCase("description")){
|
||
|
t = new StringTokenizer(substring, ":");
|
||
|
t.nextToken().trim(); // the prop we already know
|
||
|
StringBuffer tmp = new StringBuffer(t.nextToken().trim());
|
||
|
for (int i = 0; i < ATTRIBUTE_NAMES.length; i++){
|
||
|
if (tmp.toString().endsWith(ATTRIBUTE_NAMES[i])){
|
||
|
int len = ATTRIBUTE_NAMES[i].length();
|
||
|
// trim off that
|
||
|
tmp.setLength(tmp.length() - len);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
value = hansalizeIt(tmp.toString());
|
||
|
}
|
||
|
else {
|
||
|
// Single value properties like bound: true
|
||
|
t = new StringTokenizer(substring, ":\n");
|
||
|
t.nextToken().trim(); // the prop we already know
|
||
|
value = t.nextToken().trim();
|
||
|
}
|
||
|
|
||
|
// now we need to look for a match of any of the
|
||
|
// property
|
||
|
|
||
|
return value;
|
||
|
}
|
||
|
catch (Exception e){
|
||
|
return "invalidValue";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates a HashMap containing the key value pair for the parsed values
|
||
|
* of the "attributes" and "enum" tags.
|
||
|
* ie. For attribute value: visualUpdate true
|
||
|
* The HashMap will have key: visualUpdate, value: true
|
||
|
*/
|
||
|
private static HashMap getAttributeMap(String str, String delim) {
|
||
|
StringTokenizer t = new StringTokenizer(str, delim);
|
||
|
HashMap map = null;
|
||
|
String key;
|
||
|
String value;
|
||
|
|
||
|
int num = t.countTokens()/2;
|
||
|
if (num > 0) {
|
||
|
map = new HashMap();
|
||
|
for (int i = 0; i < num; i++) {
|
||
|
key = t.nextToken().trim();
|
||
|
value = t.nextToken().trim();
|
||
|
map.put(key, value);
|
||
|
}
|
||
|
}
|
||
|
return map;
|
||
|
}
|
||
|
|
||
|
// looks for extra spaces, \n hard-coded and invisible,etc
|
||
|
private static String hansalizeIt(String from){
|
||
|
char [] chars = from.toCharArray();
|
||
|
int len = chars.length;
|
||
|
int toss = 0;
|
||
|
|
||
|
// remove double spaces
|
||
|
for (int i = 0; i < len; i++){
|
||
|
if ((chars[i] == ' ')) {
|
||
|
if (i+1 < len) {
|
||
|
if ((chars[i+1] == ' ' ) || (chars[i+1] == '\n'))
|
||
|
{
|
||
|
--len;
|
||
|
System.arraycopy(chars,i+1,chars,i,len-i);
|
||
|
--i;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (chars[i] == '\n'){
|
||
|
chars[i] = ' ';
|
||
|
i -= 2;
|
||
|
}
|
||
|
|
||
|
if (chars[i] == '\\') {
|
||
|
if (i+1 < len) {
|
||
|
if (chars[i+1] == 'n'){
|
||
|
chars[i+1] = ' ';
|
||
|
--len;
|
||
|
System.arraycopy(chars,i+1, chars,i, len-i);
|
||
|
--i;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return new String(chars,0,len);
|
||
|
}
|
||
|
|
||
|
}
|