package nokogiri; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Map; import org.cyberneko.html.HTMLElements; import org.jruby.Ruby; import org.jruby.RubyClass; import org.jruby.RubyObject; import org.jruby.anno.JRubyClass; import org.jruby.anno.JRubyMethod; import org.jruby.runtime.ThreadContext; import org.jruby.runtime.builtin.IRubyObject; /** * @author Patrick Mahoney */ @JRubyClass(name="Nokogiri::HTML::ElementDescription") public class HtmlElementDescription extends RubyObject { /** * Stores memoized hash of element -> list of valid subelements. */ static protected Map> subElements; static { Map> _subElements = new HashMap>(); subElements = Collections.synchronizedMap(_subElements); } protected HTMLElements.Element element; public HtmlElementDescription(Ruby runtime, RubyClass rubyClass) { super(runtime, rubyClass); } /** * Lookup the list of sub elements of code. If not * already stored, iterate through all elements to find valid * subelements; save this list and return it. */ protected static List findSubElements(HTMLElements.Element elem) { List subs = subElements.get(elem.code); if (subs == null) { subs = new ArrayList(); /* * A bit of a hack. NekoHtml source code shows that * UNKNOWN is the highest value element. We cannot access * the list of elements directly because it's protected. */ for (short c = 0; c < HTMLElements.UNKNOWN; c++) { HTMLElements.Element maybe_sub = HTMLElements.getElement(c); if (maybe_sub.isParent(elem)) { subs.add(maybe_sub.name); } } subElements.put(elem.code, subs); } return subs; } @JRubyMethod(name="[]", meta=true) public static IRubyObject get(ThreadContext context, IRubyObject klazz, IRubyObject name) { HTMLElements.Element elem = HTMLElements.getElement(name.toString()); if (elem == HTMLElements.NO_SUCH_ELEMENT) return context.getRuntime().getNil(); HtmlElementDescription desc = new HtmlElementDescription(context.getRuntime(), (RubyClass)klazz); desc.element = elem; return desc; } @JRubyMethod() public IRubyObject name(ThreadContext context) { return context.getRuntime().newString(element.name.toLowerCase()); } @JRubyMethod(name="inline?") public IRubyObject inline_eh(ThreadContext context) { return context.getRuntime().newBoolean(element.isInline()); } @JRubyMethod(name="empty?") public IRubyObject empty_eh(ThreadContext context) { return context.getRuntime().newBoolean(element.isEmpty()); } @JRubyMethod() public IRubyObject sub_elements(ThreadContext context) { Ruby ruby = context.getRuntime(); List subs = findSubElements(element); IRubyObject[] ary = new IRubyObject[subs.size()]; for (int i = 0; i < subs.size(); ++i) { ary[i] = ruby.newString(subs.get(i)); } return ruby.newArray(ary); } }