blob: 1fc235d156c15edcb11ef223fa04e3fc653c1707 [file] [log] [blame]
/*
* Copyright 2010 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.user.client.ui;
import com.google.gwt.editor.client.IsEditor;
import com.google.gwt.editor.client.adapters.TakesValueEditor;
import com.google.gwt.event.dom.client.ChangeEvent;
import com.google.gwt.event.dom.client.ChangeHandler;
import com.google.gwt.event.logical.shared.ValueChangeEvent;
import com.google.gwt.event.logical.shared.ValueChangeHandler;
import com.google.gwt.event.shared.HandlerRegistration;
import com.google.gwt.text.shared.Renderer;
import com.google.gwt.view.client.ProvidesKey;
import com.google.gwt.view.client.SimpleKeyProvider;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* Implementation of {@link HasConstrainedValue} based on a
* {@link com.google.gwt.dom.client.SelectElement}.
* <p>
* A {@link Renderer<T>} is used to get user presentable strings to display in
* the select element.
*
* @param <T> the value type
*/
public class ValueListBox<T> extends Composite implements
HasConstrainedValue<T>, IsEditor<TakesValueEditor<T>> {
private final List<T> values = new ArrayList<T>();
private final Map<Object, Integer> valueKeyToIndex = new HashMap<Object, Integer>();
private final Renderer<T> renderer;
private final ProvidesKey<T> keyProvider;
private T value;
public ValueListBox(Renderer<T> renderer) {
this(renderer, new SimpleKeyProvider<T>());
}
public ValueListBox(Renderer<T> renderer, ProvidesKey<T> keyProvider) {
this.keyProvider = keyProvider;
this.renderer = renderer;
initWidget(new ListBox());
getListBox().addChangeHandler(new ChangeHandler() {
public void onChange(ChangeEvent event) {
int selectedIndex = getListBox().getSelectedIndex();
if (selectedIndex < 0) {
return; // Not sure why this happens during addValue
}
T newValue = values.get(selectedIndex);
setValue(newValue, true);
}
});
}
public HandlerRegistration addValueChangeHandler(ValueChangeHandler<T> handler) {
return addHandler(handler, ValueChangeEvent.getType());
}
/**
* Returns a {@link TakesValueEditor} backed by the ValueListBox.
*/
public TakesValueEditor<T> asEditor() {
return TakesValueEditor.of(this);
}
public T getValue() {
return value;
}
public void setAcceptableValues(Collection<T> newValues) {
values.clear();
valueKeyToIndex.clear();
ListBox listBox = getListBox();
listBox.clear();
for (T nextNewValue : newValues) {
addValue(nextNewValue);
}
updateListBox();
}
/**
* Set the value and display it in the select element. Add the value to the
* acceptable set if it is not already there.
*/
public void setValue(T value) {
setValue(value, false);
}
public void setValue(T value, boolean fireEvents) {
if (value == this.value || (this.value != null && this.value.equals(value))) {
return;
}
T before = this.value;
this.value = value;
updateListBox();
if (fireEvents) {
ValueChangeEvent.fireIfNotEqual(this, before, value);
}
}
private void addValue(T value) {
Object key = keyProvider.getKey(value);
if (valueKeyToIndex.containsKey(key)) {
throw new IllegalArgumentException("Duplicate value: " + value);
}
valueKeyToIndex.put(key, values.size());
values.add(value);
getListBox().addItem(renderer.render(value));
assert values.size() == getListBox().getItemCount();
}
private ListBox getListBox() {
return (ListBox) getWidget();
}
private void updateListBox() {
Object key = keyProvider.getKey(value);
Integer index = valueKeyToIndex.get(key);
if (index == null) {
addValue(value);
}
index = valueKeyToIndex.get(key);
getListBox().setSelectedIndex(index);
}
}