blob: 408541355ffe44e36cae8f210635b4c1091b2dd6 [file] [log] [blame]
/*
* 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.layout.client;
import static com.google.gwt.dom.client.Style.Unit.EM;
import static com.google.gwt.dom.client.Style.Unit.EX;
import static com.google.gwt.dom.client.Style.Unit.IN;
import static com.google.gwt.dom.client.Style.Unit.PX;
import com.google.gwt.dom.client.DivElement;
import com.google.gwt.dom.client.Document;
import com.google.gwt.dom.client.Element;
import com.google.gwt.dom.client.Style;
import com.google.gwt.dom.client.Style.Position;
import com.google.gwt.dom.client.Style.Unit;
import com.google.gwt.layout.client.Layout.Layer;
/**
* Default implementation, which works with all browsers except for IE6. It uses
* only the "top", "left", "bottom", "right", "width", and "height" CSS
* properties.
*
* Note: This implementation class has state, so {@link Layout} must create a
* new instance for each layout-parent.
*/
class LayoutImpl {
private static DivElement fixedRuler;
static {
Document doc = Document.get();
fixedRuler = createRuler(IN, IN);
doc.getBody().appendChild(fixedRuler);
}
protected static DivElement createRuler(Unit widthUnit, Unit heightUnit) {
DivElement ruler = Document.get().createDivElement();
ruler.setInnerHTML(" ");
Style style = ruler.getStyle();
style.setPosition(Position.ABSOLUTE);
style.setZIndex(-32767);
style.setLeft(-10000, PX);
style.setWidth(1, widthUnit);
style.setHeight(1, heightUnit);
return ruler;
}
protected DivElement relativeRuler;
public void attach(Element parent) {
// Do nothing. This exists only to help LayoutImplIE6 avoid memory leaks.
}
public Element attachChild(Element parent, Element child) {
DivElement container = Document.get().createDivElement();
container.appendChild(child);
container.getStyle().setProperty("position", "absolute");
container.getStyle().setProperty("overflow", "hidden");
fillParent(child);
Style style = child.getStyle();
style.setWidth(100, Unit.PCT);
style.setHeight(100, Unit.PCT);
parent.appendChild(container);
return container;
}
public void detach(Element parent) {
// Do nothing. This exists only to help LayoutImplIE6 avoid memory leaks.
}
public void fillParent(Element elem) {
Style style = elem.getStyle();
style.setPosition(Position.ABSOLUTE);
style.setLeft(0, PX);
style.setTop(0, PX);
style.setRight(0, PX);
style.setBottom(0, PX);
}
public void finalizeLayout(Element parent) {
}
public double getUnitSizeInPixels(Element parent, Unit unit, boolean vertical) {
if (unit == null) {
return 1;
}
switch (unit) {
case PCT:
return (vertical ? parent.getClientHeight() : parent.getClientWidth()) / 100.0;
case EM:
return relativeRuler.getOffsetWidth();
case EX:
return relativeRuler.getOffsetHeight();
case CM:
return fixedRuler.getOffsetWidth();
case MM:
return fixedRuler.getOffsetWidth() / 10.0;
case IN:
return fixedRuler.getOffsetWidth() / 2.54;
case PT:
return fixedRuler.getOffsetWidth() / 28.4;
case PC:
return fixedRuler.getOffsetWidth() / 2.36;
default:
case PX:
return 1;
}
}
public void initParent(Element parent) {
parent.getStyle().setProperty("position", "relative");
parent.appendChild(relativeRuler = createRuler(EM, EX));
}
public void layout(Layer layer) {
Style style = layer.container.getStyle();
style.setProperty("left", layer.setLeft
? (layer.left + layer.leftUnit.getType()) : "");
style.setProperty("top", layer.setTop
? (layer.top + layer.topUnit.getType()) : "");
style.setProperty("right", layer.setRight
? (layer.right + layer.rightUnit.getType()) : "");
style.setProperty("bottom", layer.setBottom
? (layer.bottom + layer.bottomUnit.getType()) : "");
style.setProperty("width", layer.setWidth
? (layer.width + layer.widthUnit.getType()) : "");
style.setProperty("height", layer.setHeight
? (layer.height + layer.heightUnit.getType()) : "");
}
public void removeChild(Element container, Element child) {
container.removeFromParent();
// We want this code to be resilient to the child having already been
// removed from its container (perhaps by widget code).
if (child.getParentElement() == container) {
child.removeFromParent();
}
// Cleanup child styles set by fillParent().
Style style = child.getStyle();
style.clearPosition();
style.clearLeft();
style.clearTop();
style.clearWidth();
style.clearHeight();
}
}