/*
* Copyright (c) 2010, Michael Grossmann
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of the jo-widgets.org nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*/
package org.jowidgets.impl.widgets.composed;
import java.util.Collection;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import org.jowidgets.api.controller.IContainerListener;
import org.jowidgets.api.controller.IContainerRegistry;
import org.jowidgets.api.controller.IListenerFactory;
import org.jowidgets.api.layout.ILayoutFactory;
import org.jowidgets.api.toolkit.Toolkit;
import org.jowidgets.api.widgets.IComposite;
import org.jowidgets.api.widgets.IControl;
import org.jowidgets.api.widgets.IInputComponentValidationLabel;
import org.jowidgets.api.widgets.IInputComposite;
import org.jowidgets.api.widgets.IInputControl;
import org.jowidgets.api.widgets.blueprint.ICompositeBluePrint;
import org.jowidgets.api.widgets.blueprint.IInputComponentValidationLabelBluePrint;
import org.jowidgets.api.widgets.blueprint.IScrollCompositeBluePrint;
import org.jowidgets.api.widgets.blueprint.factory.IBluePrintFactory;
import org.jowidgets.api.widgets.content.IInputContentContainer;
import org.jowidgets.api.widgets.content.IInputContentCreator;
import org.jowidgets.api.widgets.descriptor.setup.IInputCompositeSetup;
import org.jowidgets.common.color.IColorConstant;
import org.jowidgets.common.types.Dimension;
import org.jowidgets.common.types.Rectangle;
import org.jowidgets.common.widgets.controller.IComponentListener;
import org.jowidgets.common.widgets.controller.IFocusListener;
import org.jowidgets.common.widgets.controller.IInputListener;
import org.jowidgets.common.widgets.controller.IKeyListener;
import org.jowidgets.common.widgets.controller.IMouseListener;
import org.jowidgets.common.widgets.controller.IPopupDetectionListener;
import org.jowidgets.common.widgets.descriptor.IWidgetDescriptor;
import org.jowidgets.common.widgets.factory.ICustomWidgetCreator;
import org.jowidgets.common.widgets.layout.ILayoutDescriptor;
import org.jowidgets.common.widgets.layout.ILayouter;
import org.jowidgets.common.widgets.layout.MigLayoutDescriptor;
import org.jowidgets.tools.controller.InputObservable;
import org.jowidgets.tools.layout.MigLayoutFactory;
import org.jowidgets.tools.validation.ValidationCache;
import org.jowidgets.tools.validation.ValidationCache.IValidationResultCreator;
import org.jowidgets.tools.widgets.blueprint.BPF;
import org.jowidgets.tools.widgets.wrapper.ControlWrapper;
import org.jowidgets.util.Tuple;
import org.jowidgets.validation.IValidateable;
import org.jowidgets.validation.IValidationConditionListener;
import org.jowidgets.validation.IValidationResult;
import org.jowidgets.validation.IValidationResultBuilder;
import org.jowidgets.validation.IValidator;
import org.jowidgets.validation.ValidationResult;
import org.jowidgets.validation.tools.CompoundValidator;
public class InputCompositeImpl<INPUT_TYPE> extends ControlWrapper implements IInputComposite<INPUT_TYPE>, IInputContentContainer {
private final List<Tuple<String, IInputControl<?>>> inputControls;
private final List<Tuple<String, IValidateable>> validatables;
private final Set<IInputControl<?>> editedControls;
private final IInputContentCreator<INPUT_TYPE> contentCreator;
private final IComposite composite;
private final IComposite innerComposite;
private final IInputComponentValidationLabel validationLabel;
private final InputObservable inputObservable;
private final CompoundValidator<INPUT_TYPE> compoundValidator;
private final ValidationCache validationCache;
private final String missingInputHint;
private boolean editable;
public InputCompositeImpl(final IComposite composite, final IInputCompositeSetup<INPUT_TYPE> setup) {
super(composite);
this.inputControls = new LinkedList<Tuple<String, IInputControl<?>>>();
this.validatables = new LinkedList<Tuple<String, IValidateable>>();
this.editedControls = new HashSet<IInputControl<?>>();
this.inputObservable = new InputObservable();
this.contentCreator = setup.getContentCreator();
final IBluePrintFactory bpf = Toolkit.getBluePrintFactory();
this.composite = composite;
this.missingInputHint = setup.getMissingInputHint();
this.validationCache = new ValidationCache(new ValidationResultCreator());
if (setup.getValidationLabel() != null) {
final int heigth = setup.getValidationLabelHeight();
this.composite.setLayout(new MigLayoutDescriptor("0[grow]0", "0[" + heigth + "!]0[grow]0"));
}
else {
this.composite.setLayout(new MigLayoutDescriptor("0[grow]0", "0[grow]0"));
}
final IComposite innerRoot;
if (setup.isContentScrolled()) {
final IScrollCompositeBluePrint scrollCompositeBluePrint = bpf.scrollComposite();
scrollCompositeBluePrint.setBorder(setup.getContentBorder());
innerRoot = composite.add(scrollCompositeBluePrint, "growx, growy, h 0::,w 0::");
}
else {
final ICompositeBluePrint compositeBluePrint = bpf.composite();
compositeBluePrint.setBorder(setup.getContentBorder());
innerRoot = composite.add(compositeBluePrint, "growx, growy, h 0::,w 0::");
}
innerRoot.setLayout(MigLayoutFactory.growingInnerCellLayout());
innerComposite = innerRoot.add(BPF.composite(), MigLayoutFactory.GROWING_CELL_CONSTRAINTS);
this.compoundValidator = new CompoundValidator<INPUT_TYPE>();
if (setup.getValidator() != null) {
compoundValidator.addValidator(setup.getValidator());
}
contentCreator.createContent(this);
contentCreator.setValue(setup.getValue());
if (setup.getValidationLabel() != null) {
final IComposite validationContainer = this.composite.add(
0,
BPF.composite(),
MigLayoutFactory.GROWING_CELL_CONSTRAINTS + ", wrap");
validationContainer.setLayout(new MigLayoutDescriptor("[grow, 0::]", "[grow, 0::]"));
final IColorConstant validationBackground = setup.getValidationLabelBackground();
if (validationBackground != null) {
validationContainer.setBackgroundColor(validationBackground);
}
final IInputComponentValidationLabelBluePrint validationLabelBp = bpf.inputComponentValidationLabel();
validationLabelBp.setSetup(setup.getValidationLabel());
validationLabelBp.setInputComponent(this);
validationLabel = validationContainer.add(validationLabelBp, "aligny c, growx");
}
else {
validationLabel = null;
}
this.editable = setup.isEditable();
if (!setup.isEditable()) {
setEditable(false);
}
resetModificationState();
validationCache.setDirty();
if (validationLabel != null) {
validationLabel.resetValidation();
}
}
@Override
public void setEditable(final boolean editable) {
this.editable = editable;
for (final Tuple<String, IInputControl<?>> tuple : inputControls) {
tuple.getSecond().setEditable(editable);
}
}
@Override
public boolean isEditable() {
return editable;
}
@Override
public void addValidator(final IValidator<INPUT_TYPE> validator) {
compoundValidator.addValidator(validator);
}
@Override
public boolean hasModifications() {
for (final Tuple<String, IInputControl<?>> tuple : inputControls) {
if (tuple.getSecond().hasModifications()) {
return true;
}
}
return false;
}
@Override
public void resetModificationState() {
for (final Tuple<String, IInputControl<?>> tuple : inputControls) {
tuple.getSecond().resetModificationState();
}
editedControls.clear();
}
@Override
public IValidationResult validate() {
return validationCache.validate();
}
@Override
public void addValidationConditionListener(final IValidationConditionListener listener) {
validationCache.addValidationConditionListener(listener);
}
@Override
public void removeValidationConditionListener(final IValidationConditionListener listener) {
validationCache.removeValidationConditionListener(listener);
}
@Override
public void addInputListener(final IInputListener listener) {
inputObservable.addInputListener(listener);
}
@Override
public void removeInputListener(final IInputListener listener) {
inputObservable.removeInputListener(listener);
}
@Override
public void setValue(final INPUT_TYPE value) {
contentCreator.setValue(value);
resetModificationState();
validationCache.setDirty();
if (validationLabel != null) {
validationLabel.resetValidation();
}
}
@Override
public INPUT_TYPE getValue() {
return contentCreator.getValue();
}
@Override
public void setTabOrder(final Collection<? extends IControl> tabOrder) {
innerComposite.setTabOrder(tabOrder);
}
@Override
public void setTabOrder(final IControl... controls) {
innerComposite.setTabOrder(controls);
}
@Override
public void register(final String validationContext, final IValidateable validateable) {
this.validatables.add(new Tuple<String, IValidateable>(validationContext, validateable));
}
@Override
public void unregister(final String validationContext, final IValidateable validateable) {
this.validatables.remove(new Tuple<String, IValidateable>(validationContext, validateable));
}
@Override
public void setLayout(final ILayoutDescriptor layoutDescriptor) {
innerComposite.setLayout(layoutDescriptor);
}
@Override
public void layout() {
composite.layout();
}
@Override
public void layoutLater() {
composite.layoutLater();
}
@Override
public void layoutBegin() {
composite.layoutBegin();
}
@Override
public void layoutEnd() {
composite.layoutEnd();
}
@Override
public Rectangle getClientArea() {
return innerComposite.getClientArea();
}
@Override
public Dimension computeDecoratedSize(final Dimension clientAreaSize) {
return composite.computeDecoratedSize(clientAreaSize);
}
@Override
public <LAYOUT_TYPE extends ILayouter> LAYOUT_TYPE setLayout(final ILayoutFactory<LAYOUT_TYPE> layoutFactory) {
return innerComposite.setLayout(layoutFactory);
}
@Override
public List<IControl> getChildren() {
return innerComposite.getChildren();
}
@Override
public boolean remove(final IControl control) {
removeControlFromInputControls(control);
return innerComposite.remove(control);
}
@Override
public void addContainerListener(final IContainerListener listener) {
innerComposite.addContainerListener(listener);
}
@Override
public void removeContainerListener(final IContainerListener listener) {
innerComposite.removeContainerListener(listener);
}
@Override
public void addContainerRegistry(final IContainerRegistry registry) {
innerComposite.addContainerRegistry(registry);
}
@Override
public void removeContainerRegistry(final IContainerRegistry registry) {
innerComposite.removeContainerRegistry(registry);
}
@Override
public void addComponentListenerRecursive(final IListenerFactory<IComponentListener> listenerFactory) {
innerComposite.addComponentListenerRecursive(listenerFactory);
}
@Override
public void removeComponentListenerRecursive(final IListenerFactory<IComponentListener> listenerFactory) {
innerComposite.removeComponentListenerRecursive(listenerFactory);
}
@Override
public void addFocusListenerRecursive(final IListenerFactory<IFocusListener> listenerFactory) {
innerComposite.addFocusListenerRecursive(listenerFactory);
}
@Override
public void removeFocusListenerRecursive(final IListenerFactory<IFocusListener> listenerFactory) {
innerComposite.removeFocusListenerRecursive(listenerFactory);
}
@Override
public void addKeyListenerRecursive(final IListenerFactory<IKeyListener> listenerFactory) {
innerComposite.addKeyListenerRecursive(listenerFactory);
}
@Override
public void removeKeyListenerRecursive(final IListenerFactory<IKeyListener> listenerFactory) {
innerComposite.removeKeyListenerRecursive(listenerFactory);
}
@Override
public void addMouseListenerRecursive(final IListenerFactory<IMouseListener> listenerFactory) {
innerComposite.addMouseListenerRecursive(listenerFactory);
}
@Override
public void removeMouseListenerRecursive(final IListenerFactory<IMouseListener> listenerFactory) {
innerComposite.removeMouseListenerRecursive(listenerFactory);
}
@Override
public void addPopupDetectionListenerRecursive(final IListenerFactory<IPopupDetectionListener> listenerFactory) {
innerComposite.addPopupDetectionListenerRecursive(listenerFactory);
}
@Override
public void removePopupDetectionListenerRecursive(final IListenerFactory<IPopupDetectionListener> listenerFactory) {
innerComposite.removePopupDetectionListenerRecursive(listenerFactory);
}
private void removeControlFromInputControls(final IControl control) {
int index = 0;
for (final Tuple<String, IInputControl<?>> tuple : new LinkedList<Tuple<String, IInputControl<?>>>(inputControls)) {
if (tuple.getSecond() == control) {
inputControls.remove(index);
editedControls.remove(control);
return;
}
index++;
}
}
@Override
public void removeAll() {
inputControls.clear();
innerComposite.removeAll();
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final int index,
final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(index, descriptor, layoutConstraints);
afterAdd(null, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final int index,
final ICustomWidgetCreator<WIDGET_TYPE> creator,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(index, creator, layoutConstraints);
afterAdd(null, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final int index,
final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor) {
return add(index, descriptor, null);
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(descriptor, layoutConstraints);
afterAdd(null, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final ICustomWidgetCreator<WIDGET_TYPE> creator,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(creator, layoutConstraints);
afterAdd(null, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(final int index, final ICustomWidgetCreator<WIDGET_TYPE> creator) {
return add(index, creator, null);
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor) {
final WIDGET_TYPE result = innerComposite.add(descriptor);
afterAdd(null, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(final ICustomWidgetCreator<WIDGET_TYPE> creator) {
final WIDGET_TYPE result = innerComposite.add(creator);
afterAdd(null, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final String validationContext,
final int index,
final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(index, descriptor, layoutConstraints);
afterAdd(validationContext, result);
return result;
}
@Override
public <WIDGET_TYPE extends IControl> WIDGET_TYPE add(
final String validationContext,
final int index,
final ICustomWidgetCreator<WIDGET_TYPE> creator,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(index, creator, layoutConstraints);
afterAdd(validationContext, result);
return result;
}
@Override
public <WIDGET_TYPE extends IInputControl<?>> WIDGET_TYPE add(
final String validationContext,
final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(descriptor, layoutConstraints);
afterAdd(validationContext, result);
return result;
}
@Override
public <WIDGET_TYPE extends IInputControl<?>> WIDGET_TYPE add(
final String validationContext,
final ICustomWidgetCreator<WIDGET_TYPE> creator,
final Object layoutConstraints) {
final WIDGET_TYPE result = innerComposite.add(creator, layoutConstraints);
afterAdd(validationContext, result);
return result;
}
@Override
public <WIDGET_TYPE extends IInputControl<?>> WIDGET_TYPE add(
final String validationContext,
final IWidgetDescriptor<? extends WIDGET_TYPE> descriptor) {
final WIDGET_TYPE result = innerComposite.add(descriptor);
afterAdd(validationContext, result);
return result;
}
@Override
public <WIDGET_TYPE extends IInputControl<?>> WIDGET_TYPE add(
final String validationContext,
final ICustomWidgetCreator<WIDGET_TYPE> creator) {
final WIDGET_TYPE result = innerComposite.add(creator);
afterAdd(validationContext, result);
return result;
}
private void afterAdd(final String context, final IControl control) {
if (control instanceof IInputControl) {
final IInputControl<?> inputControl = (IInputControl<?>) control;
inputControls.add(new Tuple<String, IInputControl<?>>(context, inputControl));
inputControl.addValidationConditionListener(new IValidationConditionListener() {
@Override
public void validationConditionsChanged() {
validationCache.setDirty();
}
});
inputControl.addInputListener(new IInputListener() {
@Override
public void inputChanged() {
editedControls.add(inputControl);
inputObservable.fireInputChanged();
}
});
}
}
private class ValidationResultCreator implements IValidationResultCreator {
@Override
public IValidationResult createValidationResult() {
boolean hintAdded = false;
final IValidationResultBuilder builder = ValidationResult.builder();
for (final Tuple<String, IInputControl<?>> tuple : inputControls) {
final IValidationResult result = validate(tuple);
final IInputControl<?> control = tuple.getSecond();
if (control.hasModifications() || editedControls.contains(control) || missingInputHint == null) {
builder.addResult(validate(tuple));
}
else if (!hintAdded && !result.isValid()) {
builder.addInfoError(missingInputHint);
hintAdded = true;
}
}
for (final Tuple<String, IValidateable> tuple : validatables) {
builder.addResult(validate(tuple));
}
final IValidationResult compoundResult = compoundValidator.validate(getValue());
if (hasModifications() || missingInputHint == null) {
builder.addResult(compoundResult);
}
else if (!hintAdded && !compoundResult.isValid()) {
builder.addInfoError(missingInputHint);
hintAdded = true;
}
final IValidationResult result = builder.build();
return result;
}
private IValidationResult validate(final Tuple<String, ? extends IValidateable> tuple) {
final String context = tuple.getFirst();
final IValidateable inputControl = tuple.getSecond();
return inputControl.validate().withContext(context);
}
}
}