/*
* Copyright 2000-2018 Vaadin Ltd.
*
* 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.vaadin.ui;
import java.util.Collections;
import java.util.Iterator;
import java.util.Objects;
import com.vaadin.server.ErrorMessage;
import com.vaadin.server.Resource;
import com.vaadin.server.SerializableFunction;
import com.vaadin.shared.ui.ContentMode;
import com.vaadin.shared.ui.composite.CompositeState;
/**
* Composite allows creating new UI components by composition of existing
* server-side components.
*
* A composite is created by extending the Composite class and setting the
* composition root component using {@link #setCompositionRoot(Component)}.
*
*
* The composition root itself can contain more components. The advantage of
* wrapping it in a composite is that the details of the composition root, such
* as its public API, are hidden from the users of the composite.
*
*
* A composite itself does not contribute to the DOM in any way (contrary to
* {@link CustomComponent} which adds a {@code
} to the DOM.
*
*
* @author Vaadin Ltd.
* @since 8.1
*/
public class Composite extends AbstractComponent implements HasComponents {
/**
* The contained component.
*/
private Component root = null;
/**
* Constructs a new empty composite.
*
* Use {@link #setCompositionRoot(Component)} to define the contents of the
* composite.
*/
public Composite() {
}
/**
* Constructs a new composite containing the given component.
*
* @param compositionRoot
* the root of the composition component tree. It must not be
* null.
*/
public Composite(AbstractComponent compositionRoot) {
this();
Objects.requireNonNull(compositionRoot);
setCompositionRoot(compositionRoot);
}
/**
* Returns the composition root.
*
* @return the Component Composition root.
*/
protected Component getCompositionRoot() {
return root;
}
/**
* Sets the component contained in the composite.
*
* You must set the composition root to a non-null value before the
* component can be used. It cannot be changed.
*
*
* @param compositionRoot
* the root of the composition component tree.
*/
protected void setCompositionRoot(Component compositionRoot) {
if (root != null) {
throw new IllegalStateException(
"Composition root cannot be changed");
}
if (compositionRoot == null) {
throw new IllegalArgumentException(
"Composition root cannot be null");
}
// set new component
if (compositionRoot.getParent() != null) {
// If the component already has a parent, try to remove it
AbstractSingleComponentContainer.removeFromParent(compositionRoot);
}
compositionRoot.setParent(this);
root = compositionRoot;
markAsDirty();
}
/* Basic component features ------------------------------------------ */
@Override
public Iterator
iterator() {
if (getCompositionRoot() != null) {
return Collections.singletonList(getCompositionRoot()).iterator();
} else {
return Collections. emptyList().iterator();
}
}
@Override
protected CompositeState getState() {
return (CompositeState) super.getState();
}
@Override
protected CompositeState getState(boolean markAsDirty) {
return (CompositeState) super.getState(markAsDirty);
}
@Override
public void beforeClientResponse(boolean initial) {
super.beforeClientResponse(initial);
if (getCompositionRoot() == null) {
throw new IllegalStateException(
"A composite must always have a composition root");
}
}
@Override
public String getStyleName() {
Component root = getCompositionRoot();
return root == null ? "" : root.getStyleName();
}
@Override
public void setStyleName(String style) {
getRootOrThrow().setStyleName(style);
}
@Override
public void setStyleName(String style, boolean add) {
getRootAbstractComponentOrThrow().setStyleName(style, add);
}
@Override
public void addStyleName(String style) {
getRootOrThrow().addStyleName(style);
}
@Override
public void removeStyleName(String style) {
getRootOrThrow().removeStyleName(style);
}
@Override
public String getPrimaryStyleName() {
return getRootAbstractComponentPropertyOrNull(
AbstractComponent::getPrimaryStyleName);
}
@Override
public void setPrimaryStyleName(String style) {
getRootOrThrow().setPrimaryStyleName(style);
}
private Component getRootOrThrow() {
Component root = getCompositionRoot();
if (root == null) {
throw new IllegalStateException(
"Composition root has not been set");
}
return root;
}
private AbstractComponent getRootAbstractComponentOrThrow() {
Component root = getRootOrThrow();
if (!(root instanceof AbstractComponent)) {
throw new IllegalStateException(
"Composition root is not AbstractComponent");
}
return (AbstractComponent) root;
}
private T getRootPropertyOrNull(
SerializableFunction getter) {
Component root = getCompositionRoot();
return root == null ? null : getter.apply(root);
}
private T getRootAbstractComponentPropertyOrNull(
SerializableFunction getter) {
Component root = getCompositionRoot();
if (root instanceof AbstractComponent) {
return getter.apply((AbstractComponent) root);
}
return null;
}
@Override
public void setEnabled(boolean enabled) {
getRootOrThrow().setEnabled(enabled);
}
@Override
public boolean isEnabled() {
return getRootOrThrow().isEnabled();
}
@Override
public float getWidth() {
return getRootOrThrow().getWidth();
}
@Override
public float getHeight() {
return getRootOrThrow().getHeight();
}
@Override
public Unit getWidthUnits() {
return getRootOrThrow().getWidthUnits();
}
@Override
public Unit getHeightUnits() {
return getRootOrThrow().getHeightUnits();
}
@Override
public void setHeight(String height) {
getRootOrThrow().setHeight(height);
}
@Override
public void setWidth(float width, Unit unit) {
getRootOrThrow().setWidth(width, unit);
}
@Override
public void setHeight(float height, Unit unit) {
getRootOrThrow().setHeight(height, unit);
}
@Override
public void setWidth(String width) {
getRootOrThrow().setWidth(width);
}
@Override
public void setSizeFull() {
getRootOrThrow().setSizeFull();
}
@Override
public void setSizeUndefined() {
getRootOrThrow().setSizeUndefined();
}
@Override
public void setWidthUndefined() {
getRootOrThrow().setWidthUndefined();
}
@Override
public void setHeightUndefined() {
getRootOrThrow().setHeightUndefined();
}
@Override
public void setId(String id) {
getRootOrThrow().setId(id);
}
@Override
public String getId() {
return getRootPropertyOrNull(Component::getId);
}
@Override
public void setDebugId(String id) {
getRootAbstractComponentOrThrow().setDebugId(id);
}
@Override
public String getDebugId() {
return getRootAbstractComponentPropertyOrNull(
AbstractComponent::getDebugId);
}
@Override
public String getCaption() {
return getRootPropertyOrNull(Component::getCaption);
}
@Override
public void setCaption(String caption) {
getRootOrThrow().setCaption(caption);
}
@Override
public void setCaptionAsHtml(boolean captionAsHtml) {
getRootAbstractComponentOrThrow().setCaptionAsHtml(captionAsHtml);
}
@Override
public boolean isCaptionAsHtml() {
return getRootAbstractComponentPropertyOrNull(
AbstractComponent::isCaptionAsHtml);
}
@Override
public Resource getIcon() {
return getRootPropertyOrNull(Component::getIcon);
}
@Override
public void setIcon(Resource icon) {
getRootOrThrow().setIcon(icon);
}
@Override
public String getDescription() {
return getRootOrThrow().getDescription();
}
@Override
public void setDescription(String description) {
getRootAbstractComponentOrThrow().setDescription(description);
}
@Override
public void setDescription(String description, ContentMode mode) {
getRootAbstractComponentOrThrow().setDescription(description, mode);
}
@Override
public ErrorMessage getErrorMessage() {
return getRootAbstractComponentPropertyOrNull(
AbstractComponent::getErrorMessage);
}
@Override
public ErrorMessage getComponentError() {
return getRootAbstractComponentPropertyOrNull(
AbstractComponent::getComponentError);
}
@Override
public void setComponentError(ErrorMessage componentError) {
getRootAbstractComponentOrThrow().setComponentError(componentError);
}
}