/* * @(#)AbstractSubWriter.java 1.24 99/05/28 * * Copyright 1997-1999 by Sun Microsystems, Inc., * 901 San Antonio Road, Palo Alto, California, 94303, U.S.A. * All rights reserved. * * This software is the confidential and proprietary information * of Sun Microsystems, Inc. ("Confidential Information"). You * shall not disclose such Confidential Information and shall use * it only in accordance with the terms of the license agreement * you entered into with Sun. */ package com.sun.tools.doclets.standard; import com.sun.tools.doclets.*; import com.sun.javadoc.*; import java.util.*; import java.lang.reflect.Modifier; /** * * @author Robert Field * @author Atul M Dambalkar */ public abstract class AbstractSubWriter { protected final SubWriterHolderWriter writer; public List inhmembers; public List supclasses; protected AbstractSubWriter(SubWriterHolderWriter writer) { this.writer = writer; } /*** abstracts ***/ public abstract ProgramElementDoc[] members(ClassDoc cd); public abstract void printSummaryLabel(ClassDoc cd); public abstract void printInheritedSummaryLabel(ClassDoc cd); public abstract void printSummaryAnchor(ClassDoc cd); public abstract void printInheritedSummaryAnchor(ClassDoc cd); protected abstract void printSummaryType(ProgramElementDoc member); protected abstract void printSummaryLink(ClassDoc cd, ProgramElementDoc member); protected abstract void printInheritedSummaryLink(ClassDoc cd, ProgramElementDoc member); protected abstract void printHeader(ClassDoc cd); protected abstract void printBodyHtmlEnd(ClassDoc cd); protected abstract void printMember(ProgramElementDoc elem); protected abstract void printDeprecatedLink(ProgramElementDoc member); protected abstract void printNavSummaryLink(ClassDoc cd, boolean link); protected abstract void printNavDetailLink(boolean link); /*** ***/ protected void print(String str) { writer.print(str); writer.displayLength += str.length(); } protected void print(char ch) { writer.print(ch); writer.displayLength++; } protected void bold(String str) { writer.bold(str); writer.displayLength += str.length(); } protected void printTypeLinkNoDimension(Type type) { ClassDoc cd = type.asClassDoc(); if (cd == null) { print(type.typeName()); } else { writer.printClassLink(cd); } } protected void printTypeLink(Type type) { printTypeLinkNoDimension(type); print(type.dimension()); } /** * Return a string describing the access modifier flags. * Don't include native or synchronized. * * The modifier names are returned in canonical order, as * specified by The Java Language Specification. */ protected String modifierString(MemberDoc member) { int ms = member.modifierSpecifier(); int no = Modifier.NATIVE | Modifier.SYNCHRONIZED; return Modifier.toString(ms & ~no); } protected String typeString(MemberDoc member) { String type = ""; if (member instanceof MethodDoc) { type = ((MethodDoc)member).returnType().toString(); } else if (member instanceof FieldDoc) { type = ((FieldDoc)member).type().toString(); } return type; } protected void printModifiers(MemberDoc member) { String mod; mod = modifierString(member); if(mod.length() > 0) { print(mod); print(' '); } } protected void printTypedName(Type type, String name) { if (type != null) { printTypeLink(type); } if(name.length() > 0) { writer.space(); writer.print(name); } } protected String makeSpace(int len) { if (len <= 0) { return ""; } StringBuffer sb = new StringBuffer(len); for(int i = 0; i < len; i++) { sb.append(' '); } return sb.toString(); } /** * Print 'static' if static and type link. */ protected void printStaticAndType(boolean isStatic, Type type) { writer.printTypeSummaryHeader(); if (isStatic) { print("static"); } writer.space(); if (type != null) { printTypeLink(type); } writer.printTypeSummaryFooter(); } protected void printModifierAndType(ProgramElementDoc member, Type type) { writer.printTypeSummaryHeader(); printModifier(member); if (type == null) { if (member.isClass()) { print("class"); } else { print("interface"); } } else { printTypeLink(type); } writer.printTypeSummaryFooter(); } protected void printModifier(ProgramElementDoc member) { if (member.isProtected()) { print("protected "); } else if (member.isPrivate()) { print("private "); } else if (!member.isPublic()) { // Package private writer.printText("doclet.Package_private"); print(" "); } if (member.isMethod() && ((MethodDoc)member).isAbstract()) { print("abstract "); } if (member.isStatic()) { print("static"); } writer.space(); } protected void printComment(ProgramElementDoc member) { if (member.inlineTags().length > 0) { writer.dd(); writer.printInlineComment(member); } } protected void printTags(ProgramElementDoc member) { Tag[] since = member.tags("since"); if (member.seeTags().length + since.length > 0) { writer.dd(); writer.dl(); writer.printSeeTags(member); writer.printSinceTag(member); writer.dlEnd(); writer.ddEnd(); } } protected String name(ProgramElementDoc member) { return member.name(); } protected void printDeprecated(ProgramElementDoc member) { Tag[] deprs = member.tags("deprecated"); if (deprs.length > 0) { writer.dd(); writer.boldText("doclet.Deprecated"); writer.space(); writer.printInlineDeprecatedComment(deprs[0]); writer.p(); } else { printDeprecatedClassComment(member); } } protected void printDeprecatedClassComment(ProgramElementDoc member) { Tag[] deprs = member.containingClass().tags("deprecated"); if (deprs.length > 0) { writer.dd(); writer.boldText("doclet.Deprecated"); writer.space(); } } protected void printHead(MemberDoc member) { writer.h3(); writer.print(member.name()); writer.h3End(); } protected void printFullComment(ProgramElementDoc member) { writer.dl(); printDeprecated(member); printCommentAndTags(member); writer.dlEnd(); } protected void printCommentAndTags(ProgramElementDoc member) { printComment(member); printTags(member); } /** * Forward to containing writer */ public void printSummaryHeader(ClassDoc cd) { writer.printSummaryHeader(this, cd); } /** * Forward to containing writer */ public void printInheritedSummaryHeader(ClassDoc cd) { writer.printInheritedSummaryHeader(this, cd); } /** * Forward to containing writer */ public void printInheritedSummaryFooter(ClassDoc cd) { writer.printInheritedSummaryFooter(this, cd); } /** * Forward to containing writer */ public void printSummaryFooter(ClassDoc cd) { writer.printSummaryFooter(this, cd); } /** * Forward to containing writer */ public void printSummaryMember(ClassDoc cd, ProgramElementDoc member) { writer.printSummaryMember(this, cd, member); } /** * Forward to containing writer */ public void printInheritedSummaryMember(ClassDoc cd, ProgramElementDoc member) { writer.printInheritedSummaryMember(this, cd, member); } public void printMembersSummary(ClassDoc cd) { ProgramElementDoc[] members = members(cd); if (members.length > 0) { Arrays.sort(members); printSummaryHeader(cd); for (int i = 0; i < members.length; ++i) { printSummaryMember(cd, members[i]); } printSummaryFooter(cd); } } public void printInheritedMembersSummary(ClassDoc cd) { inhmembers = new ArrayList(); supclasses = new ArrayList(); printRecursivelyInheritedMembersSummary(cd); } public void printRecursivelyInheritedMembersSummary(ClassDoc cd) { if (cd.isClass()) { ClassDoc icd = cd.superclass(); while (icd != null) { printInheritedMembersInfo(icd); icd = icd.superclass(); } printInheritedMembersSummaryForImplInterfaces(cd); } else { // this is an interface printInheritedMembersSummaryForImplInterfaces(cd); } } public void printInheritedMembersSummaryForImplInterfaces(ClassDoc cd) { ClassDoc[] iin = cd.interfaces(); for (int i = 0; i < iin.length; i++) { printInheritedMembersInfo(iin[i]); } for (int i = 0; i < iin.length; i++) { printRecursivelyInheritedMembersSummary(iin[i]); } } protected void printInheritedMembersInfo(ClassDoc icd) { if (supclasses.contains(icd)) { return; // already traversed this class. } supclasses.add(icd); ProgramElementDoc[] members = members(icd); members = checkForDuplicateMembers(members); if (members.length > 0) { Arrays.sort(members); printInheritedSummaryHeader(icd); printInheritedSummaryMember(icd, members[0]); for (int i = 1; i < members.length; ++i) { print(", "); writer.println(' '); printInheritedSummaryMember(icd, members[i]); } printInheritedSummaryFooter(icd); } } protected ProgramElementDoc[] checkForDuplicateMembers( ProgramElementDoc[] members) { List memberlist = new ArrayList(); for (int i = 0; i < members.length; i++) { ProgramElementDoc member = members[i]; boolean found = false; for (int j = 0; j < inhmembers.size(); j++) { ProgramElementDoc inhmember = (ProgramElementDoc)inhmembers.get(j); if (isEqual(member, inhmember)) { found = true; break; } } if (!found) { memberlist.add(member); inhmembers.add(member); } } return (ProgramElementDoc[])memberlist. toArray(new ProgramElementDoc[memberlist.size()]); } public static boolean isEqual(ProgramElementDoc doc1, ProgramElementDoc doc2) { if (doc1 instanceof ExecutableMemberDoc && doc2 instanceof ExecutableMemberDoc) { ExecutableMemberDoc ed1 = (ExecutableMemberDoc)doc1; ExecutableMemberDoc ed2 = (ExecutableMemberDoc)doc2; return ed1.name().equals(ed2.name()) && ed1.signature().equals(ed2.signature()); } else { return doc1.name().equals(doc2.name()); } } public void printMembers(ClassDoc cd) { ProgramElementDoc[] members = members(cd); if (members.length > 0) { printHeader(cd); for (int i = 0; i < members.length; ++i) { if (i > 0) { writer.printMemberHeader(); } writer.println(""); printMember(members[i]); writer.printMemberFooter(); } printBodyHtmlEnd(cd); } } /** * Generate the code for listing the deprecated APIs. Create the table * format for listing the API. Call methods from the sub-class to complete * the generation. */ protected void printDeprecatedAPI(List deprmembers, String headingKey) { if (deprmembers.size() > 0) { writer.tableIndexSummary(); writer.tableHeaderStart("#CCCCFF"); writer.boldText(headingKey); writer.tableHeaderEnd(); for (int i = 0; i < deprmembers.size(); i++) { ProgramElementDoc member =(ProgramElementDoc)deprmembers.get(i); ClassDoc cd = member.containingClass(); writer.trBgcolorStyle("white", "TableRowColor"); writer.summaryRow(0); printDeprecatedLink(member); writer.br(); writer.printNbsps(); writer.printInlineDeprecatedComment(member.tags("deprecated")[0]); writer.space(); writer.summaryRowEnd(); writer.trEnd(); } writer.tableEnd(); writer.space(); writer.p(); } } /** * Print use info. */ protected void printUseInfo(Object mems, String heading) { if (mems == null) { return; } List members = (List)mems; if (members.size() > 0) { writer.tableIndexSummary(); writer.tableUseInfoHeaderStart("#CCCCFF"); writer.print(heading); writer.tableHeaderEnd(); for (Iterator it = members.iterator(); it.hasNext(); ) { printSummaryMember(null, (ProgramElementDoc)it.next()); } writer.tableEnd(); writer.space(); writer.p(); } } protected void navSummaryLink(ClassDoc cd) { ProgramElementDoc[] members = members(cd); if (members.length > 0) { printNavSummaryLink(null, true); return; } else { ClassDoc icd = cd.superclass(); while (icd != null) { ProgramElementDoc[] inhmembers = members(icd); if (inhmembers.length > 0) { printNavSummaryLink(icd, true); return; } icd = icd.superclass(); } } printNavSummaryLink(null, false); } protected void navDetailLink(ClassDoc cd) { ProgramElementDoc[] members = members(cd); printNavDetailLink(members.length > 0? true: false); } protected ProgramElementDoc[] eligibleMembers(ProgramElementDoc[] members) { if (!Standard.configuration().nodeprecated) { return members; } List list = new ArrayList(); for (int i = 0; i < members.length; i++) { if (members[i].tags("deprecated").length == 0) { list.add(members[i]); } } return (ProgramElementDoc[])list. toArray(new ProgramElementDoc[list.size()]); } protected void serialWarning(String key, String a1, String a2) { if (Standard.configuration().serialwarn) { Standard.configuration().standardmessage.warning(key, a1, a2); } } }