Adds TabLayoutPanelParser.
Review: http://gwt-code-reviews.appspot.com/81802

git-svn-id: https://google-web-toolkit.googlecode.com/svn/trunk@6399 8db76d5a-ed1c-0410-87a9-c151d255dfc7
diff --git a/user/src/com/google/gwt/uibinder/parsers/TabLayoutPanelParser.java b/user/src/com/google/gwt/uibinder/parsers/TabLayoutPanelParser.java
new file mode 100644
index 0000000..18dc393
--- /dev/null
+++ b/user/src/com/google/gwt/uibinder/parsers/TabLayoutPanelParser.java
@@ -0,0 +1,151 @@
+/*
+ * Copyright 2009 Google Inc.
+ * 
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ * 
+ * http://www.apache.org/licenses/LICENSE-2.0
+ * 
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.google.gwt.uibinder.parsers;
+
+import com.google.gwt.core.ext.UnableToCompleteException;
+import com.google.gwt.core.ext.typeinfo.JClassType;
+import com.google.gwt.dom.client.Style.Unit;
+import com.google.gwt.uibinder.rebind.UiBinderWriter;
+import com.google.gwt.uibinder.rebind.XMLElement;
+import com.google.gwt.user.client.ui.TabLayoutPanel;
+
+/**
+ * Parses {@link TabLayoutPanel} widgets.
+ */
+public class TabLayoutPanelParser implements ElementParser {
+
+  private static class Children {
+    XMLElement body;
+    XMLElement header;
+    XMLElement customHeader;
+  }
+
+  private static final String CUSTOM = "customHeader";
+  private static final String HEADER = "header";
+  private static final String TAB = "tab";
+
+  public void parse(XMLElement panelElem, String fieldName, JClassType type,
+      UiBinderWriter writer) throws UnableToCompleteException {
+    // TabLayoutPanel requires tabBar size and unit ctor args.
+    double size = panelElem.consumeDoubleAttribute("barHeight");
+    Unit unit = panelElem.consumeEnumAttribute("barUnit", Unit.class);
+
+    String enumName = DockLayoutPanelParser.getFullyQualifiedEnumName(unit);
+    JClassType tlpType = writer.getOracle().findType(
+        TabLayoutPanel.class.getName());
+    writer.setFieldInitializerAsConstructor(fieldName, tlpType,
+        Double.toString(size), enumName);
+
+    // Parse children.
+    for (XMLElement tabElem : panelElem.consumeChildElements()) {
+      // Get the tab element.
+      if (!isElementType(panelElem, tabElem, TAB)) {
+        writer.die("In %s, only <%s:%s> children are allowed.", panelElem,
+            panelElem.getPrefix(), TAB);
+      }
+
+      // Find all the children of the <tab>.
+      Children children = findChildren(tabElem, writer);
+
+      // Parse the child widget.
+      if (children.body == null) {
+        writer.die("%s must have a child widget", tabElem);
+      }
+      if (!writer.isWidgetElement(children.body)) {
+        writer.die("In %s, %s must be a widget", tabElem, children.body);
+      }
+      String childFieldName = writer.parseElementToField(children.body);
+
+      // Parse the header.
+      if (children.header != null) {
+        HtmlInterpreter htmlInt = HtmlInterpreter.newInterpreterForUiObject(
+            writer, fieldName);
+        String html = children.header.consumeInnerHtml(htmlInt);
+        writer.addStatement("%s.add(%s, \"%s\", true);", fieldName,
+            childFieldName, html);
+      } else if (children.customHeader != null) {
+        XMLElement headerElement =
+          children.customHeader.consumeSingleChildElement();
+
+        if (!writer.isWidgetElement(headerElement)) {
+          writer.die("In %s of %s, %s is not a widget", children.customHeader,
+              tabElem, headerElement);
+        }
+
+        String headerField = writer.parseElementToField(headerElement);
+        writer.addStatement("%s.add(%s, %s);", fieldName, childFieldName,
+            headerField);
+      } else {
+        // Neither a header or customHeader.
+        writer.die("%1$s requires either a <%2$s:%3$s> or <%2$s:%4$s>",
+            tabElem, tabElem.getPrefix(), HEADER, CUSTOM);
+      }
+    }
+  }
+
+  private Children findChildren(final XMLElement elem,
+      final UiBinderWriter writer) throws UnableToCompleteException {
+    final Children children = new Children();
+
+    elem.consumeChildElements(new XMLElement.Interpreter<Boolean>() {
+      public Boolean interpretElement(XMLElement child)
+          throws UnableToCompleteException {
+
+        if (hasTag(child, HEADER)) {
+          assertFirstHeader();
+          children.header = child;
+          return true;
+        }
+
+        if (hasTag(child, CUSTOM)) {
+          assertFirstHeader();
+          children.customHeader = child;
+          return true;
+        }
+
+        // Must be the body, then
+        if (null != children.body) {
+          writer.die("In %s, may have only one body element", elem);
+        }
+
+        children.body = child;
+        return true;
+      }
+
+      void assertFirstHeader() throws UnableToCompleteException {
+        if ((null != children.header) && (null != children.customHeader)) {
+          writer.die("In %1$s, may have only one %2$s:header "
+              + "or %2$s:customHeader", elem, elem.getPrefix());
+        }
+      }
+
+      private boolean hasTag(XMLElement child, final String attribute) {
+        return rightNamespace(child) && child.getLocalName().equals(attribute);
+      }
+
+      private boolean rightNamespace(XMLElement child) {
+        return child.getNamespaceUri().equals(elem.getNamespaceUri());
+      }
+    });
+
+    return children;
+  }
+
+  private boolean isElementType(XMLElement parent, XMLElement child, String type) {
+    return child.getNamespaceUri().equals(parent.getNamespaceUri())
+        && type.equals(child.getLocalName());
+  }
+}
diff --git a/user/src/com/google/gwt/uibinder/rebind/UiBinderWriter.java b/user/src/com/google/gwt/uibinder/rebind/UiBinderWriter.java
index 6f406cb..0b676ba 100644
--- a/user/src/com/google/gwt/uibinder/rebind/UiBinderWriter.java
+++ b/user/src/com/google/gwt/uibinder/rebind/UiBinderWriter.java
@@ -1117,6 +1117,7 @@
     addWidgetParser("CustomButton");
     addWidgetParser("DockLayoutPanel");
     addWidgetParser("StackLayoutPanel");
+    addWidgetParser("TabLayoutPanel");
 
     addAttributeParser("boolean",
         "com.google.gwt.uibinder.parsers.BooleanAttributeParser");
diff --git a/user/src/com/google/gwt/user/client/ui/TabLayoutPanel.java b/user/src/com/google/gwt/user/client/ui/TabLayoutPanel.java
index fc05226..81fdfee 100644
--- a/user/src/com/google/gwt/user/client/ui/TabLayoutPanel.java
+++ b/user/src/com/google/gwt/user/client/ui/TabLayoutPanel.java
@@ -63,6 +63,8 @@
     RequiresResize, ProvidesResize, IndexedPanel,
     HasBeforeSelectionHandlers<Integer>, HasSelectionHandlers<Integer> {
 
+  private static final int BIG_ENOUGH_TO_NOT_WRAP = 16384;
+
   private static class Tab extends SimplePanel {
     private Element anchor;
 
@@ -106,20 +108,20 @@
   private WidgetCollection children = new WidgetCollection(this);
   private FlowPanel tabBar = new FlowPanel();
   private ArrayList<Tab> tabs = new ArrayList<Tab>();
-  private final double tabBarSize;
-  private final Unit tabBarUnit;
+  private final double barHeight;
+  private final Unit barUnit;
   private LayoutPanel panel;
   private int selectedIndex = -1;
 
   /**
    * Creates an empty tab panel. 
    * 
-   * @param tabBarSize the size of the tab bar
-   * @param tabBarUnit the unit in which the tab bar size is specified
+   * @param barHeight the size of the tab bar
+   * @param barUnit the unit in which the tab bar size is specified
    */
-  public TabLayoutPanel(double tabBarSize, Unit tabBarUnit) {
-    this.tabBarSize = tabBarSize;
-    this.tabBarUnit = tabBarUnit;
+  public TabLayoutPanel(double barHeight, Unit barUnit) {
+    this.barHeight = barHeight;
+    this.barUnit = barUnit;
 
     panel = new LayoutPanel();
     initWidget(panel);
@@ -127,11 +129,15 @@
     panel.add(tabBar);
     Layer layer = panel.getLayer(tabBar);
     layer.setLeftRight(0, Unit.PX, 0, Unit.PX);
-    layer.setTopHeight(0, Unit.PX, tabBarSize, tabBarUnit);
+    layer.setTopHeight(0, Unit.PX, barHeight, barUnit);
     panel.layout();
 
     panel.getLayer(tabBar).setChildVerticalPosition(Alignment.END);
 
+    // Make the tab bar extremely wide so that tabs themselves never wrap.
+    // (Its layout container is overflow:hidden)
+    tabBar.getElement().getStyle().setWidth(BIG_ENOUGH_TO_NOT_WRAP, Unit.PX);
+
     tabBar.setStyleName("gwt-TabLayoutPanelTabs");
     setStyleName("gwt-TabLayoutPanel");
   }
@@ -145,7 +151,7 @@
    * moved to the right-most index.
    * 
    * @param child the widget to be added
-   * @param tabText the text to be shown on its tab
+   * @param text the text to be shown on its tab
    */
   public void add(Widget child, String text) {
     insert(child, text, getWidgetCount());
@@ -156,11 +162,11 @@
    * moved to the right-most index.
    * 
    * @param child the widget to be added
-   * @param tabText the text to be shown on its tab
+   * @param text the text to be shown on its tab
    * @param asHtml <code>true</code> to treat the specified text as HTML
    */
-  public void add(Widget w, String text, boolean asHtml) {
-    insert(w, text, asHtml, getWidgetCount());
+  public void add(Widget child, String text, boolean asHtml) {
+    insert(child, text, asHtml, getWidgetCount());
   }
 
   /**
@@ -241,11 +247,10 @@
    * be moved to the requested index.
    * 
    * @param child the widget to be added
-   * @param tab the widget to be placed in the associated tab
    * @param beforeIndex the index before which it will be inserted
    */
-  public void insert(Widget w, int beforeIndex) {
-    insert(w, "", beforeIndex);
+  public void insert(Widget child, int beforeIndex) {
+    insert(child, "", beforeIndex);
   }
 
   /**
@@ -253,18 +258,18 @@
    * moved to the requested index.
    * 
    * @param child the widget to be added
-   * @param tabText the text to be shown on its tab
+   * @param text the text to be shown on its tab
    * @param asHtml <code>true</code> to treat the specified text as HTML
    * @param beforeIndex the index before which it will be inserted
    */
-  public void insert(Widget w, String text, boolean asHtml, int beforeIndex) {
+  public void insert(Widget child, String text, boolean asHtml, int beforeIndex) {
     Widget contents;
     if (asHtml) {
       contents = new HTML(text);
     } else {
       contents = new Label(text);
     }
-    insert(w, contents, beforeIndex);
+    insert(child, contents, beforeIndex);
   }
 
   /**
@@ -272,7 +277,7 @@
    * moved to the requested index.
    * 
    * @param child the widget to be added
-   * @param tabText the text to be shown on its tab
+   * @param text the text to be shown on its tab
    * @param beforeIndex the index before which it will be inserted
    */
   public void insert(Widget child, String text, int beforeIndex) {
@@ -343,8 +348,8 @@
 
     // Fire the before selection event, giving the recipients a chance to
     // cancel the selection.
-    BeforeSelectionEvent<Integer> event = BeforeSelectionEvent
-        .fire(this, index);
+    BeforeSelectionEvent<Integer> event =
+      BeforeSelectionEvent.fire(this, index);
     if ((event != null) && event.isCanceled()) {
       return;
     }
@@ -384,9 +389,9 @@
    * @param index the index of the tab whose HTML is to be set
    * @param html the tab's new HTML contents
    */
-  public void setTabHTML(int index, String text) {
+  public void setTabHTML(int index, String html) {
     checkIndex(index);
-    tabs.get(index).setWidget(new HTML(text));
+    tabs.get(index).setWidget(new HTML(html));
   }
 
   /**
@@ -442,7 +447,7 @@
   private void layoutChild(Widget child) {
     Layer layer = panel.getLayer(child);
     layer.setLeftRight(0, Unit.PX, 0, Unit.PX);
-    layer.setTopBottom(tabBarSize, tabBarUnit, 0, Unit.PX);
+    layer.setTopBottom(barHeight, barUnit, 0, Unit.PX);
     layer.getContainerElement().getStyle().setVisibility(Visibility.HIDDEN);
     panel.layout();
   }