summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/com/itmill/toolkit/demo/BrowserDemo.java80
-rw-r--r--src/com/itmill/toolkit/demo/BufferedComponents.java90
-rw-r--r--src/com/itmill/toolkit/demo/CachingDemo.java69
-rw-r--r--src/com/itmill/toolkit/demo/Calc.java4
-rw-r--r--src/com/itmill/toolkit/demo/FilterSelect.java86
-rw-r--r--src/com/itmill/toolkit/demo/HelloWorld.java34
-rw-r--r--src/com/itmill/toolkit/demo/KeyboardShortcut.java147
-rw-r--r--src/com/itmill/toolkit/demo/NotificationDemo.java92
-rw-r--r--src/com/itmill/toolkit/demo/PortletDemo.java153
-rw-r--r--src/com/itmill/toolkit/demo/SelectDemo.java81
-rw-r--r--src/com/itmill/toolkit/demo/SimpleAddressBook.java137
-rw-r--r--src/com/itmill/toolkit/demo/TableDemo.java182
-rw-r--r--src/com/itmill/toolkit/demo/ToolkitTunesLayout.java142
-rw-r--r--src/com/itmill/toolkit/demo/WindowedDemos.java97
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/Coverflow.java89
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.html149
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.java109
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/CoverflowWidgetSet.gwt.xml8
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/client/CoverflowWidgetSet.java35
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/client/ui/ICoverflow.java336
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/Cover.as392
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/Coverflow.as598
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/EventHandler.as16
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/coverflowflash.mxml34
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/DistortImage.as306
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/SandyPoint.as17
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/Triangle.as20
-rw-r--r--src/com/itmill/toolkit/demo/coverflow/gwt/public/coverflowflash.swfbin0 -> 176877 bytes
-rw-r--r--src/com/itmill/toolkit/demo/m-bullet-blue.gifbin8417 -> 0 bytes
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/AddressBookApplication.java244
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/data/Person.java117
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/data/PersonContainer.java91
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/data/SearchFilter.java41
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/HelpWindow.java17
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/ListView.java12
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/NavigationTree.java28
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonForm.java176
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonList.java45
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SearchView.java99
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SharingOptions.java39
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/validators/EmailValidator.java22
-rw-r--r--src/com/itmill/toolkit/demo/tutorial/addressbook/validators/PostalCodeValidator.java22
42 files changed, 3303 insertions, 1153 deletions
diff --git a/src/com/itmill/toolkit/demo/BrowserDemo.java b/src/com/itmill/toolkit/demo/BrowserDemo.java
deleted file mode 100644
index 91756b09fd..0000000000
--- a/src/com/itmill/toolkit/demo/BrowserDemo.java
+++ /dev/null
@@ -1,80 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import com.itmill.toolkit.data.Property.ValueChangeEvent;
-import com.itmill.toolkit.terminal.ExternalResource;
-import com.itmill.toolkit.ui.Embedded;
-import com.itmill.toolkit.ui.Select;
-import com.itmill.toolkit.ui.VerticalLayout;
-import com.itmill.toolkit.ui.Window;
-
-/**
- * Demonstrates the use of Embedded and "suggesting" Select by creating a simple
- * web-browser. Note: does not check for recursion.
- *
- * @author IT Mill Ltd.
- * @see com.itmill.toolkit.ui.Window
- */
-public class BrowserDemo extends com.itmill.toolkit.Application implements
- Select.ValueChangeListener {
-
- // Default URL to open.
- private static final String DEFAULT_URL = "http://www.itmill.com";
-
- // The embedded page
- Embedded emb = new Embedded();
-
- @Override
- public void init() {
- // Create and set main window
- final Window browser = new Window("IT Mill Browser");
- setMainWindow(browser);
-
- // Use the expand layout to allow one component to use as much
- // space as
- // possible.
- final VerticalLayout exl = new VerticalLayout();
- browser.setLayout(exl);
- exl.setSizeFull();
-
- // create the address combobox
- final Select select = new Select();
- // allow input
- select.setNewItemsAllowed(true);
- // no empty selection
- select.setNullSelectionAllowed(false);
- select.setWidth("100%");
- // no 'go' -button clicking necessary
- select.setImmediate(true);
- // add some pre-configured URLs
- select.addItem(DEFAULT_URL);
- select.addItem("http://www.google.com");
- select.addItem("http://toolkit.itmill.com/demo");
- // add to layout
- exl.addComponent(select);
- // add listener and select initial URL
- select.addListener(this);
- select.setValue(DEFAULT_URL);
-
- // configure the embedded and add to layout
- emb.setType(Embedded.TYPE_BROWSER);
- emb.setSizeFull();
- exl.addComponent(emb);
- // make the embedded as large as possible
- exl.setExpandRatio(emb, 1.0F);
-
- }
-
- public void valueChange(ValueChangeEvent event) {
- final String url = (String) event.getProperty().getValue();
- if (url != null) {
- // the selected url has changed, let's go there
- emb.setSource(new ExternalResource(url));
- }
-
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/BufferedComponents.java b/src/com/itmill/toolkit/demo/BufferedComponents.java
deleted file mode 100644
index e66ba6481a..0000000000
--- a/src/com/itmill/toolkit/demo/BufferedComponents.java
+++ /dev/null
@@ -1,90 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import com.itmill.toolkit.Application;
-import com.itmill.toolkit.data.util.MethodProperty;
-import com.itmill.toolkit.data.util.ObjectProperty;
-import com.itmill.toolkit.ui.Button;
-import com.itmill.toolkit.ui.Label;
-import com.itmill.toolkit.ui.TextField;
-import com.itmill.toolkit.ui.Window;
-import com.itmill.toolkit.ui.Button.ClickEvent;
-import com.itmill.toolkit.ui.Window.Notification;
-
-public class BufferedComponents extends Application {
-
- private ObjectProperty property;
-
- private TextField text;
-
- @Override
- public void init() {
-
- final Window w = new Window("Buffered UI components demo");
- addWindow(w);
-
- // Create property
- final Float floatValue = new Float(1.0f);
- property = new ObjectProperty(floatValue);
-
- // Textfield
- text = new TextField("TextField (Buffered, using ObjectProperty)",
- property);
- text.setImmediate(true);
- text.setWriteThrough(false);
- w.addComponent(text);
-
- // Property state
- final Label propertyState = new Label(property);
- propertyState.setCaption("Property (data source) state");
- w.addComponent(propertyState);
-
- // Button state
- final Label textState = new Label(text);
- textState.setCaption("TextField state");
- w.addComponent(textState);
-
- // Button to change the property
- w.addComponent(new Button("increase property value",
- new Button.ClickListener() {
- public void buttonClick(ClickEvent event) {
- final Float currentValue = (Float) property.getValue();
- property.setValue(new Float(
- currentValue.floatValue() + 1.0));
- }
- }));
-
- // Buffering
- w.addComponent(new Button("Write through enabled", new MethodProperty(
- text, "writeThrough")));
- w.addComponent(new Button("discard", new Button.ClickListener() {
- public void buttonClick(ClickEvent event) {
- text.discard();
- }
- }));
- Button commit = new Button("commit", new Button.ClickListener() {
- public void buttonClick(ClickEvent event) {
- try {
- text.commit();
- w.showNotification("Committed " + property
- + " to datasource.");
- } catch (Throwable e) {
- w.showNotification("Error committing an invalid value: "
- + text, Notification.TYPE_WARNING_MESSAGE);
- }
- }
- });
- w.addComponent(commit);
-
- // Restart button for application
- // (easier debugging when you dont have to restart the server to
- // make
- // code changes)
- final Button restart = new Button("restart", this, "close");
- restart.addStyleName(Button.STYLE_LINK);
- w.addComponent(restart);
- }
-}
diff --git a/src/com/itmill/toolkit/demo/CachingDemo.java b/src/com/itmill/toolkit/demo/CachingDemo.java
deleted file mode 100644
index 820ccca7e2..0000000000
--- a/src/com/itmill/toolkit/demo/CachingDemo.java
+++ /dev/null
@@ -1,69 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import com.itmill.toolkit.terminal.PaintException;
-import com.itmill.toolkit.terminal.PaintTarget;
-import com.itmill.toolkit.ui.Label;
-import com.itmill.toolkit.ui.Layout;
-import com.itmill.toolkit.ui.OrderedLayout;
-import com.itmill.toolkit.ui.TabSheet;
-import com.itmill.toolkit.ui.Window;
-
-/**
- * This example is a (simple) demonstration of client-side caching. The content
- * in one tab is intentionally made very slow to produce server-side. When the
- * user changes to this tab for the first time, there will be a 3 second wait
- * before the content shows up, but the second time it shows up immediately
- * since the content has not changed and is cached client-side.
- *
- * @author IT Mill Ltd.
- */
-public class CachingDemo extends com.itmill.toolkit.Application {
-
- @Override
- public void init() {
-
- final Window main = new Window("Client-side caching example");
- setMainWindow(main);
-
- setTheme("example");
-
- final TabSheet ts = new TabSheet();
- main.addComponent(ts);
-
- Layout layout = new OrderedLayout();
- layout.setMargin(true);
- Label l = new Label(
- "This is a normal label, quick to render.<br/>The second tab will be slow to render the first time, after that it will be as quick as this one.");
- l.setCaption("A normal label");
- l.setContentMode(Label.CONTENT_XHTML);
- layout.addComponent(l);
-
- ts.addTab(layout, "Normal", null);
-
- layout = new OrderedLayout();
- layout.setMargin(true);
- l = new Label(
- "The first time you change to this tab, this label is very slow to produce (server-side).<br/> However, it will seem fast the second time you change to this tab, because it has not changed and is cached client-side.") {
- @Override
- public void paintContent(PaintTarget target) throws PaintException {
- try {
- Thread.sleep(3000);
- } catch (final Exception e) {
- // IGNORED
- }
- super.paintContent(target);
- }
-
- };
- l.setCaption("A slow label");
- l.setContentMode(Label.CONTENT_XHTML);
- layout.addComponent(l);
- ts.addTab(layout, "Slow", null);
-
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/Calc.java b/src/com/itmill/toolkit/demo/Calc.java
index bc9e898a52..0955f34d1f 100644
--- a/src/com/itmill/toolkit/demo/Calc.java
+++ b/src/com/itmill/toolkit/demo/Calc.java
@@ -1,7 +1,3 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
package com.itmill.toolkit.demo;
import com.itmill.toolkit.ui.Button;
diff --git a/src/com/itmill/toolkit/demo/FilterSelect.java b/src/com/itmill/toolkit/demo/FilterSelect.java
deleted file mode 100644
index f4a9472fcd..0000000000
--- a/src/com/itmill/toolkit/demo/FilterSelect.java
+++ /dev/null
@@ -1,86 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import com.itmill.toolkit.ui.OrderedLayout;
-import com.itmill.toolkit.ui.Panel;
-import com.itmill.toolkit.ui.Select;
-import com.itmill.toolkit.ui.Window;
-import com.itmill.toolkit.ui.AbstractSelect.Filtering;
-
-/**
- * The classic "hello, world!" example for IT Mill Toolkit. The class simply
- * implements the abstract {@link com.itmill.toolkit.Application#init() init()}
- * method in which it creates a Window and adds a Label to it.
- *
- * @author IT Mill Ltd.
- * @see com.itmill.toolkit.Application
- * @see com.itmill.toolkit.ui.Window
- * @see com.itmill.toolkit.ui.Label
- */
-public class FilterSelect extends com.itmill.toolkit.Application {
-
- private static final String[] firstnames = new String[] { "John", "Mary",
- "Joe", "Sarah", "Jeff", "Jane", "Peter", "Marc", "Robert", "Paula",
- "Lenny", "Kenny", "Nathan", "Nicole", "Laura", "Jos", "Josie",
- "Linus" };
-
- private static final String[] lastnames = new String[] { "Torvalds",
- "Smith", "Adams", "Black", "Wilson", "Richards", "Thompson",
- "McGoff", "Halas", "Jones", "Beck", "Sheridan", "Picard", "Hill",
- "Fielding", "Einstein" };
-
- /**
- * The initialization method that is the only requirement for inheriting the
- * com.itmill.toolkit.service.Application class. It will be automatically
- * called by the framework when a user accesses the application.
- */
- @Override
- public void init() {
-
- /*
- * - Create new window for the application - Give the window a visible
- * title - Set the window to be the main window of the application
- */
- final Window main = new Window("Filter select demo");
- setMainWindow(main);
-
- // default filterin (Starts with)
- final Select s1 = new Select();
- for (int i = 0; i < 105; i++) {
- s1
- .addItem(firstnames[(int) (Math.random() * (firstnames.length - 1))]
- + " "
- + lastnames[(int) (Math.random() * (lastnames.length - 1))]);
- }
- s1.setImmediate(true);
-
- // contains filter
- final Select s2 = new Select();
- for (int i = 0; i < 500; i++) {
- s2
- .addItem(firstnames[(int) (Math.random() * (firstnames.length - 1))]
- + " "
- + lastnames[(int) (Math.random() * (lastnames.length - 1))]);
- }
- s2.setFilteringMode(Filtering.FILTERINGMODE_CONTAINS);
-
- // Add selects to UI using ordered layout and panels
- final OrderedLayout orderedLayout = new OrderedLayout(
- OrderedLayout.ORIENTATION_HORIZONTAL);
-
- final Panel panel1 = new Panel("Select with default filter");
- final Panel panel2 = new Panel("Select with contains filter");
-
- panel1.addComponent(s1);
- panel2.addComponent(s2);
-
- orderedLayout.addComponent(panel1);
- orderedLayout.addComponent(panel2);
- main.addComponent(orderedLayout);
-
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/HelloWorld.java b/src/com/itmill/toolkit/demo/HelloWorld.java
index bcbf447d09..d77ce235f2 100644
--- a/src/com/itmill/toolkit/demo/HelloWorld.java
+++ b/src/com/itmill/toolkit/demo/HelloWorld.java
@@ -1,48 +1,22 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
package com.itmill.toolkit.demo;
import com.itmill.toolkit.ui.Label;
import com.itmill.toolkit.ui.Window;
-/**
- * The classic "hello, world!" example for IT Mill Toolkit. The class simply
- * implements the abstract {@link com.itmill.toolkit.Application#init() init()}
- * method in which it creates a Window and adds a Label to it.
- *
- * @author IT Mill Ltd.
- * @see com.itmill.toolkit.Application
- * @see com.itmill.toolkit.ui.Window
- * @see com.itmill.toolkit.ui.Label
- */
public class HelloWorld extends com.itmill.toolkit.Application {
/**
- * The initialization method that is the only requirement for inheriting the
- * com.itmill.toolkit.service.Application class. It will be automatically
- * called by the framework when a user accesses the application.
+ * Init is invoked on application load (when a user accesses the application
+ * for the first time).
*/
@Override
public void init() {
- /*
- * - Create new window for the application - Give the window a visible
- * title - Set the window to be the main window of the application
- */
+ // Main window is the primary browser window
final Window main = new Window("Hello window");
setMainWindow(main);
- /*
- * - Create a label with the classic text - Add the label to the main
- * window
- */
+ // "Hello world" text is added to window as a Label component
main.addComponent(new Label("Hello World!"));
-
- /*
- * And that's it! The framework will display the main window and its
- * contents when the application is accessed with the terminal.
- */
}
}
diff --git a/src/com/itmill/toolkit/demo/KeyboardShortcut.java b/src/com/itmill/toolkit/demo/KeyboardShortcut.java
deleted file mode 100644
index 7d3a3a1977..0000000000
--- a/src/com/itmill/toolkit/demo/KeyboardShortcut.java
+++ /dev/null
@@ -1,147 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import java.util.Date;
-
-import com.itmill.toolkit.Application;
-import com.itmill.toolkit.event.Action;
-import com.itmill.toolkit.event.ShortcutAction;
-import com.itmill.toolkit.event.Action.Handler;
-import com.itmill.toolkit.ui.Button;
-import com.itmill.toolkit.ui.HorizontalLayout;
-import com.itmill.toolkit.ui.Label;
-import com.itmill.toolkit.ui.Panel;
-import com.itmill.toolkit.ui.TextField;
-import com.itmill.toolkit.ui.VerticalLayout;
-import com.itmill.toolkit.ui.Window;
-
-/**
- * Test application for KeyboardShortcuts
- */
-public class KeyboardShortcut extends Application implements Handler {
-
- private VerticalLayout loki;
-
- private final Label instructions = new Label(
- "<p>Keyboard shortcuts is a must have feature for applications in a "
- + "daily use. In IT Mill toolkit shortcuts are binded to "
- + "Panel and its subclasses like Windows (most common place)."
- + "</p>"
- + "<p>Browsers reserve some keyboard combinations for their own"
- + " actions, so all combinations cannot be used in web "
- + "applications. (see our article on <a href=\"http://www.itmill"
- + ".com/articles/Keybindings_in_Web_Browsers.htm\">"
- + "www.itmill.com)</a></p>"
- + "<p>Focus must be inside web application (eg. not in address "
- + "bar) for shortcuts to work. By default app element is focused.</p>"
- + "<strong>Shortcuts used in this example:</strong> "
- + "<br/>ESC restarts program, ctrl-shift-a (Button A), "
- + "ctrl-shift-z (Button Z), ctrl-shift-x (Button X)",
- Label.CONTENT_XHTML);
-
- private final Action ACTION_A = new ShortcutAction("Button a action",
- ShortcutAction.KeyCode.A, new int[] {
- ShortcutAction.ModifierKey.CTRL,
- ShortcutAction.ModifierKey.SHIFT });
-
- private final Action ACTION_Z = new ShortcutAction("Button z action",
- ShortcutAction.KeyCode.Z, new int[] {
- ShortcutAction.ModifierKey.CTRL,
- ShortcutAction.ModifierKey.SHIFT });
-
- private final Action ACTION_X = new ShortcutAction("Button x action",
- ShortcutAction.KeyCode.X, new int[] {
- ShortcutAction.ModifierKey.CTRL,
- ShortcutAction.ModifierKey.SHIFT });
-
- private final Action ACTION_RESTART = new ShortcutAction("Restart ",
- ShortcutAction.KeyCode.ESCAPE, null);
-
- private final Action[] actions = new Action[] { ACTION_A, ACTION_Z,
- ACTION_X, ACTION_RESTART };
-
- private TextField f;
-
- @Override
- public void init() {
-
- final Window w = new Window("Keyboard shortcuts demo");
- final VerticalLayout main = new VerticalLayout();
- main.setMargin(true);
- main.setSpacing(true);
- setMainWindow(w);
- w.setLayout(main);
-
- final Panel p = new Panel("Test application for shortcut actions");
- p.addComponent(instructions);
-
- final HorizontalLayout buttons = new HorizontalLayout();
-
- // Restart button
- final Button close = new Button("restart", this, "close");
- final Button a = new Button("Button A", this, "actionAHandler");
- final Button z = new Button("Button Z", this, "actionZHandler");
- final Button x = new Button("Button X", this, "actionXHandler");
- f = new TextField();
-
- buttons.addComponent(close);
-
- buttons.addComponent(a);
- buttons.addComponent(z);
- buttons.addComponent(x);
- buttons.addComponent(f);
- p.addComponent(buttons);
-
- main.addComponent(p);
-
- loki = new VerticalLayout();
- main.addComponent(loki);
- main.setExpandRatio(loki, 1.0f);
-
- w.addActionHandler(this);
-
- }
-
- public Action[] getActions(Object target, Object sender) {
- return actions;
- }
-
- public void handleAction(Action action, Object sender, Object target) {
- if (action == ACTION_A) {
- actionAHandler();
- }
- if (action == ACTION_Z) {
- actionZHandler();
- }
- if (action == ACTION_X) {
- actionXHandler();
- }
- if (action == ACTION_RESTART) {
- actionRestartHandler();
- }
- }
-
- public void actionAHandler() {
- log("Button A handler fired");
- }
-
- public void actionZHandler() {
- log("Button Z handler fired");
- }
-
- public void actionXHandler() {
- log("Button X handler fired");
- }
-
- public void actionRestartHandler() {
- close();
- }
-
- public void log(String s) {
- loki.addComponentAsFirst(new Label(new Date() + " : " + s));
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/NotificationDemo.java b/src/com/itmill/toolkit/demo/NotificationDemo.java
deleted file mode 100644
index 8abe65ace8..0000000000
--- a/src/com/itmill/toolkit/demo/NotificationDemo.java
+++ /dev/null
@@ -1,92 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import com.itmill.toolkit.data.Item;
-import com.itmill.toolkit.ui.AbstractSelect;
-import com.itmill.toolkit.ui.Button;
-import com.itmill.toolkit.ui.NativeSelect;
-import com.itmill.toolkit.ui.RichTextArea;
-import com.itmill.toolkit.ui.TextField;
-import com.itmill.toolkit.ui.Window;
-import com.itmill.toolkit.ui.Button.ClickEvent;
-import com.itmill.toolkit.ui.Button.ClickListener;
-
-/**
- * Demonstrates the use of Notifications.
- *
- * @author IT Mill Ltd.
- * @see com.itmill.toolkit.ui.Window
- */
-public class NotificationDemo extends com.itmill.toolkit.Application {
-
- // Dropdown select for notification type, using the native dropdown
- NativeSelect type;
- // Textfield for the notification caption
- TextField caption;
- // Textfield for the notification content
- TextField message;
-
- /**
- * The initialization method that is the only requirement for inheriting the
- * com.itmill.toolkit.service.Application class. It will be automatically
- * called by the framework when a user accesses the application.
- */
- @Override
- public void init() {
-
- // Create new window for the application and give the window a visible.
- final Window main = new Window("Notification demo");
- // set as main window
- setMainWindow(main);
-
- // Create the 'type' dropdown select.
- type = new NativeSelect("Notification type");
- // no empty selection allowed
- type.setNullSelectionAllowed(false);
- // we want a different caption than the value
- type.addContainerProperty("caption", String.class, null);
- type.setItemCaptionMode(AbstractSelect.ITEM_CAPTION_MODE_PROPERTY);
- type.setItemCaptionPropertyId("caption");
- // add some content (items) using the Container API
- Item i = type.addItem(new Integer(
- Window.Notification.TYPE_HUMANIZED_MESSAGE));
- i.getItemProperty("caption").setValue("Humanized message");
- i = type.addItem(new Integer(Window.Notification.TYPE_WARNING_MESSAGE));
- i.getItemProperty("caption").setValue("Warning message");
- i = type.addItem(new Integer(Window.Notification.TYPE_ERROR_MESSAGE));
- i.getItemProperty("caption").setValue("Error message");
- i = type
- .addItem(new Integer(Window.Notification.TYPE_TRAY_NOTIFICATION));
- i.getItemProperty("caption").setValue("Tray notification");
- // set the initially selected item
- type.setValue(new Integer(Window.Notification.TYPE_HUMANIZED_MESSAGE));
- main.addComponent(type); // add to layout
-
- // Notification caption
- caption = new TextField("Caption");
- caption.setColumns(20);
- caption.setValue("Brown Fox!");
- main.addComponent(caption);
-
- // Notification message
- message = new RichTextArea();
- message.setCaption("Message");
- message.setValue("A quick one jumped over the lazy dog.");
- main.addComponent(message); // add to layout
-
- // Button to show the notification
- final Button b = new Button("Show notification", new ClickListener() {
- // this is an inline ClickListener
- public void buttonClick(ClickEvent event) {
- // show the notification
- getMainWindow().showNotification((String) caption.getValue(),
- (String) message.getValue(),
- ((Integer) type.getValue()).intValue());
- }
- });
- main.addComponent(b); // add button to layout
- }
-}
diff --git a/src/com/itmill/toolkit/demo/PortletDemo.java b/src/com/itmill/toolkit/demo/PortletDemo.java
deleted file mode 100644
index 57d8027795..0000000000
--- a/src/com/itmill/toolkit/demo/PortletDemo.java
+++ /dev/null
@@ -1,153 +0,0 @@
-/**
- *
- */
-package com.itmill.toolkit.demo;
-
-import java.util.Iterator;
-import java.util.Map;
-
-import javax.portlet.ActionRequest;
-import javax.portlet.ActionResponse;
-import javax.portlet.PortletMode;
-import javax.portlet.PortletRequest;
-import javax.portlet.PortletURL;
-import javax.portlet.RenderRequest;
-import javax.portlet.RenderResponse;
-import javax.portlet.WindowState;
-
-import com.itmill.toolkit.Application;
-import com.itmill.toolkit.terminal.ExternalResource;
-import com.itmill.toolkit.terminal.gwt.server.PortletApplicationContext;
-import com.itmill.toolkit.terminal.gwt.server.PortletApplicationContext.PortletListener;
-import com.itmill.toolkit.ui.Label;
-import com.itmill.toolkit.ui.Link;
-import com.itmill.toolkit.ui.TextField;
-import com.itmill.toolkit.ui.Window;
-import com.itmill.toolkit.ui.Window.Notification;
-
-/**
- * @author marc
- *
- */
-public class PortletDemo extends Application {
-
- Window main = new Window();
- TextField tf = new TextField("Some value");
- Label userInfo = new Label();
- Link portletEdit = new Link();
- Link portletMax = new Link();
- Link someAction = null;
-
- @Override
- public void init() {
- main = new Window();
- setMainWindow(main);
-
- userInfo.setCaption("User info");
- userInfo.setContentMode(Label.CONTENT_PREFORMATTED);
- main.addComponent(userInfo);
-
- tf.setEnabled(false);
- tf.setImmediate(true);
- main.addComponent(tf);
-
- portletEdit.setEnabled(false);
- main.addComponent(portletEdit);
- portletMax.setEnabled(false);
- main.addComponent(portletMax);
-
- if (getContext() instanceof PortletApplicationContext) {
- PortletApplicationContext ctx = (PortletApplicationContext) getContext();
- ctx.addPortletListener(this, new DemoPortletListener());
- } else {
- getMainWindow().showNotification("Not inited via Portal!",
- Notification.TYPE_ERROR_MESSAGE);
- }
-
- }
-
- private class DemoPortletListener implements PortletListener {
-
- public void handleActionRequest(ActionRequest request,
- ActionResponse response) {
-
- main.addComponent(new Label("Action received"));
-
- }
-
- public void handleRenderRequest(RenderRequest request,
- RenderResponse response) {
- // Portlet up-and-running, enable stuff
- portletEdit.setEnabled(true);
- portletMax.setEnabled(true);
-
- // Editable if we're in editmode
- tf.setEnabled((request.getPortletMode() == PortletMode.EDIT));
-
- // Show notification about current mode and state
- getMainWindow().showNotification(
- "Portlet status",
- "Mode: " + request.getPortletMode() + " State: "
- + request.getWindowState(),
- Notification.TYPE_WARNING_MESSAGE);
-
- // Display current user info
- Map uinfo = (Map) request.getAttribute(PortletRequest.USER_INFO);
- if (uinfo != null) {
- String s = "";
- for (Iterator it = uinfo.keySet().iterator(); it.hasNext();) {
- Object key = it.next();
- Object val = uinfo.get(key);
- s += key + ": " + val + "\n";
- }
- if (request.isUserInRole("administrator")) {
- s += "(administrator)";
- }
- userInfo.setValue(s);
- } else {
- userInfo.setValue("-");
- }
-
- // Create Edit/Done link (actionUrl)
- PortletURL url = response.createActionURL();
- try {
- url
- .setPortletMode((request.getPortletMode() == PortletMode.VIEW ? PortletMode.EDIT
- : PortletMode.VIEW));
- portletEdit.setResource(new ExternalResource(url.toString()));
- portletEdit
- .setCaption((request.getPortletMode() == PortletMode.VIEW ? "Edit"
- : "Done"));
- } catch (Exception e) {
- portletEdit.setEnabled(false);
- }
- // Create Maximize/Normal link (actionUrl)
- url = response.createActionURL();
- try {
- url
- .setWindowState((request.getWindowState() == WindowState.NORMAL ? WindowState.MAXIMIZED
- : WindowState.NORMAL));
- portletMax.setResource(new ExternalResource(url.toString()));
- portletMax
- .setCaption((request.getWindowState() == WindowState.NORMAL ? "Maximize"
- : "Back to normal"));
- } catch (Exception e) {
- portletMax.setEnabled(false);
- }
-
- if (someAction == null) {
- url = response.createActionURL();
- try {
- someAction = new Link("An action", new ExternalResource(url
- .toString()));
- main.addComponent(someAction);
- } catch (Exception e) {
- // Oops
- System.err.println("Could not create someAction: " + e);
- }
-
- }
-
- }
- }
-}
diff --git a/src/com/itmill/toolkit/demo/SelectDemo.java b/src/com/itmill/toolkit/demo/SelectDemo.java
deleted file mode 100644
index ec0ddcb647..0000000000
--- a/src/com/itmill/toolkit/demo/SelectDemo.java
+++ /dev/null
@@ -1,81 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import java.sql.SQLException;
-
-import com.itmill.toolkit.data.util.QueryContainer;
-import com.itmill.toolkit.demo.util.SampleDatabase;
-import com.itmill.toolkit.ui.Label;
-import com.itmill.toolkit.ui.Panel;
-import com.itmill.toolkit.ui.Select;
-import com.itmill.toolkit.ui.Window;
-import com.itmill.toolkit.ui.AbstractSelect.Filtering;
-
-/**
- * This example demonstrates what is lazy loading feature on Select component.
- * Demo Uses similar concepts to QueryContainerDemo.
- *
- * @author IT Mill Ltd.
- * @since 4.0.0
- *
- */
-public class SelectDemo extends com.itmill.toolkit.Application {
-
- // Select component where SQL rows are attached (using QueryContainer)
- private final Select select = new Select();
-
- private final Select lazySelect = new Select();
-
- // Database provided with sample data
- private SampleDatabase sampleDatabase;
-
- /**
- * Initialize Application. Demo components are added to main window.
- */
- @Override
- public void init() {
- final Window main = new Window("Select demo");
- setMainWindow(main);
-
- // Main window contains heading, table, select and tree
- final Panel panel = new Panel("Select demo (a.k.a Google Suggests)");
- panel.addComponent(lazySelect);
- panel.addComponent(new Label("<hr />", Label.CONTENT_XHTML));
- panel.addComponent(select);
- main.addComponent(panel);
-
- // create demo database
- sampleDatabase = new SampleDatabase();
-
- initSelects();
- }
-
- private void initSelects() {
- // init select
- select.setCaption("All employees default functionality.");
- select.setItemCaptionPropertyId("WORKER");
- // populate Toolkit select component with test SQL table rows
- try {
- final QueryContainer qc = new QueryContainer(
- "SELECT ID, UNIT||', '||LASTNAME||' '||FIRSTNAME"
- + " AS WORKER FROM employee ORDER BY WORKER",
- sampleDatabase.getConnection());
- select.setContainerDataSource(qc);
- } catch (final SQLException e) {
- e.printStackTrace();
- }
-
- // init lazySelect
- lazySelect.setCaption("All employees with lazy loading "
- + "(a.k.a Google Suggests) activated.");
- lazySelect.setItemCaptionPropertyId("WORKER");
- lazySelect.setFilteringMode(Filtering.FILTERINGMODE_CONTAINS);
-
- // use same datasource as select object uses
- lazySelect.setContainerDataSource(select.getContainerDataSource());
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/SimpleAddressBook.java b/src/com/itmill/toolkit/demo/SimpleAddressBook.java
new file mode 100644
index 0000000000..f43c987036
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/SimpleAddressBook.java
@@ -0,0 +1,137 @@
+package com.itmill.toolkit.demo;
+
+import com.itmill.toolkit.Application;
+import com.itmill.toolkit.data.Property;
+import com.itmill.toolkit.data.Property.ValueChangeEvent;
+import com.itmill.toolkit.data.util.IndexedContainer;
+import com.itmill.toolkit.ui.Button;
+import com.itmill.toolkit.ui.Form;
+import com.itmill.toolkit.ui.HorizontalLayout;
+import com.itmill.toolkit.ui.SplitPanel;
+import com.itmill.toolkit.ui.Table;
+import com.itmill.toolkit.ui.TextField;
+import com.itmill.toolkit.ui.VerticalLayout;
+import com.itmill.toolkit.ui.Window;
+import com.itmill.toolkit.ui.Button.ClickEvent;
+
+public class SimpleAddressBook extends Application {
+
+ private static String[] fields = { "First Name", "Last Name", "Company",
+ "Mobile Phone", "Work Phone", "Home Phone", "Work Email",
+ "Home Email", "Street", "Zip", "City", "State", "Country" };
+ private static String[] visibleCols = new String[] { "Last Name", "First Name",
+ "Company" };
+
+ private Table contactList = new Table();
+ private Form contactEditor = new Form();
+ private HorizontalLayout bottomLeftCorner = new HorizontalLayout();
+ private Button contactRemovalButton;
+ private IndexedContainer addressBookData = createDummyData();
+
+ public void init() {
+ initLayout();
+ initContactAddRemoveButtons();
+ initAddressList();
+ initFilteringControls();
+ }
+
+ private void initLayout() {
+ SplitPanel splitPanel = new SplitPanel(
+ SplitPanel.ORIENTATION_HORIZONTAL);
+ setMainWindow(new Window("Address Book", splitPanel));
+ VerticalLayout left = new VerticalLayout();
+ left.setSizeFull();
+ left.addComponent(contactList);
+ contactList.setSizeFull();
+ left.setExpandRatio(contactList, 1);
+ splitPanel.addComponent(left);
+ splitPanel.addComponent(contactEditor);
+ contactEditor.setSizeFull();
+ contactEditor.getLayout().setMargin(true);
+ contactEditor.setImmediate(true);
+ bottomLeftCorner.setWidth("100%");
+ left.addComponent(bottomLeftCorner);
+ }
+
+ private void initContactAddRemoveButtons() {
+ // New item button
+ bottomLeftCorner.addComponent(new Button("+", new Button.ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ Object id = contactList.addItem();
+ contactList.setValue(id);
+ }
+ }));
+
+ // Remove item button
+ contactRemovalButton = new Button("-", new Button.ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ contactList.removeItem(contactList.getValue());
+ contactList.select(null);
+ }
+ });
+ contactRemovalButton.setVisible(false);
+ bottomLeftCorner.addComponent(contactRemovalButton);
+ }
+
+ private String[] initAddressList() {
+ contactList.setContainerDataSource(addressBookData);
+ contactList.setVisibleColumns(visibleCols);
+ contactList.setSelectable(true);
+ contactList.setImmediate(true);
+ contactList.addListener(new Property.ValueChangeListener() {
+ public void valueChange(ValueChangeEvent event) {
+ Object id = contactList.getValue();
+ contactEditor.setItemDataSource(id == null ? null : contactList
+ .getItem(id));
+ contactRemovalButton.setVisible(id != null);
+ }
+ });
+ return visibleCols;
+ }
+
+ private void initFilteringControls() {
+ for (final String pn : visibleCols) {
+ final TextField sf = new TextField();
+ bottomLeftCorner.addComponent(sf);
+ sf.setWidth("100%");
+ sf.setValue(pn);
+ sf.setImmediate(true);
+ bottomLeftCorner.setExpandRatio(sf, 1);
+ sf.addListener(new Property.ValueChangeListener() {
+ public void valueChange(ValueChangeEvent event) {
+ addressBookData.removeContainerFilters(pn);
+ if (sf.toString().length() > 0 && !pn.equals(sf.toString()))
+ addressBookData.addContainerFilter(pn, sf.toString(),
+ true, false);
+ getMainWindow().showNotification(
+ "" + addressBookData.size() + " matches found");
+ }
+ });
+ }
+ }
+ private static IndexedContainer createDummyData() {
+
+ String[] fnames = { "Peter", "Alice", "Joshua", "Mike", "Olivia",
+ "Nina", "Alex", "Rita", "Dan", "Umberto", "Henrik", "Rene",
+ "Lisa", "Marge" };
+ String[] lnames = { "Smith", "Gordon", "Simpson", "Brown", "Clavel",
+ "Simons", "Verne", "Scott", "Allison", "Gates", "Rowling",
+ "Barks", "Ross", "Schneider", "Tate" };
+
+ IndexedContainer ic = new IndexedContainer();
+
+ for (String p : fields)
+ ic.addContainerProperty(p, String.class, "");
+
+ for (int i = 0; i < 1000; i++) {
+ Object id = ic.addItem();
+ ic.getContainerProperty(id, "First Name").setValue(
+ fnames[(int) (fnames.length * Math.random())]);
+ ic.getContainerProperty(id, "Last Name").setValue(
+ lnames[(int) (lnames.length * Math.random())]);
+ }
+
+ return ic;
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/TableDemo.java b/src/com/itmill/toolkit/demo/TableDemo.java
deleted file mode 100644
index c5eee75959..0000000000
--- a/src/com/itmill/toolkit/demo/TableDemo.java
+++ /dev/null
@@ -1,182 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import java.sql.SQLException;
-
-import com.itmill.toolkit.data.util.QueryContainer;
-import com.itmill.toolkit.demo.util.SampleDatabase;
-import com.itmill.toolkit.event.Action;
-import com.itmill.toolkit.ui.Button;
-import com.itmill.toolkit.ui.Label;
-import com.itmill.toolkit.ui.OrderedLayout;
-import com.itmill.toolkit.ui.Table;
-import com.itmill.toolkit.ui.Window;
-
-/**
- * Similar to QueryContainerDemo
- *
- * @author IT Mill Ltd.
- * @since 4.0.0
- *
- */
-public class TableDemo extends com.itmill.toolkit.Application implements
- Action.Handler {
-
- private static final String ACTION_DESCRIPTION = "Use right mouse button to initiate "
- + "actions menu.<br />Note: on Opera use meta key "
- + "and left mouse button.";
-
- private static final String TABLE_CAPTION = SampleDatabase.ROWCOUNT
- + " dynamically loaded rows from example SQL table";
-
- // Table component where SQL rows are attached (using QueryContainer)
- private final Table table = new Table();
-
- // Label which displays last performed action against table row
- private final Label tableLastAction = new Label(
- "No action selected for table.");
-
- // Database provided with sample data
- private SampleDatabase sampleDatabase;
-
- // Example Actions for table
- private final Action ACTION1 = new Action("Upload");
-
- private final Action ACTION2 = new Action("Download");
-
- private final Action ACTION3 = new Action("Show history");
-
- private final Action[] actions = new Action[] { ACTION1, ACTION2, ACTION3 };
-
- // Button which is used to disable or enable table
- // note: when button click event occurs, tableEnabler() method is called
- private final Button tableEnabler = new Button("Disable table", this,
- "tableEnabler");
-
- // Button which is used to hide or show table
- // note: when button click event occurs, tableVisibility() method is called
- private final Button tableVisibility = new Button("Hide table", this,
- "tableVisibility");
-
- // Button which is used to hide or show table
- // note: when button click event occurs, tableVisibility() method is called
- private final Button tableCaption = new Button("Hide caption", this,
- "tableCaption");
-
- /**
- * Initialize Application. Demo components are added to main window.
- */
- @Override
- public void init() {
- final Window main = new Window("Table demo");
- setMainWindow(main);
-
- // create demo database
- sampleDatabase = new SampleDatabase();
-
- // Main window contains heading, two buttons, table and label
- main
- .addComponent(new Label(
- "<h1>Table demo</h1>"
- + "<b>Rows are loaded from the server as they are needed.<br />"
- + "Try scrolling the table to see it in action.</b><br />"
- + ACTION_DESCRIPTION, Label.CONTENT_XHTML));
- final OrderedLayout layout = new OrderedLayout(
- OrderedLayout.ORIENTATION_HORIZONTAL);
- layout.addComponent(tableVisibility);
- layout.addComponent(tableEnabler);
- layout.addComponent(tableCaption);
- main.addComponent(layout);
- main.addComponent(table);
- main.addComponent(tableLastAction);
-
- // initialize demo components
- initTable();
- }
-
- /**
- * Populates table component with all rows from employee table.
- *
- */
- private void initTable() {
- // init table
- table.setCaption(TABLE_CAPTION);
- table.setPageLength(10);
- table.setSelectable(true);
- table.setRowHeaderMode(Table.ROW_HEADER_MODE_INDEX);
- table.setColumnCollapsingAllowed(true);
- table.setColumnReorderingAllowed(true);
- table.setSelectable(true);
- // this class handles table actions (see handleActions method below)
- table.addActionHandler(this);
- table.setDescription(ACTION_DESCRIPTION);
-
- // populate Toolkit table component with test SQL table rows
- try {
- final QueryContainer qc = new QueryContainer(
- "SELECT * FROM employee", sampleDatabase.getConnection());
- table.setContainerDataSource(qc);
- } catch (final SQLException e) {
- e.printStackTrace();
- }
- // define which columns should be visible on Table component
- table.setVisibleColumns(new Object[] { "FIRSTNAME", "LASTNAME",
- "TITLE", "UNIT" });
- table.setItemCaptionPropertyId("ID");
- }
-
- public void tableVisibility() {
- if (table.isVisible()) {
- tableVisibility.setCaption("Show table");
- table.setVisible(false);
- tableEnabler.setEnabled(false);
- tableCaption.setEnabled(false);
- } else {
- tableVisibility.setCaption("Hide table");
- table.setVisible(true);
- tableEnabler.setEnabled(true);
- tableCaption.setEnabled(true);
- }
- }
-
- public void tableEnabler() {
- if (table.isEnabled()) {
- tableEnabler.setCaption("Enable table");
- table.setEnabled(false);
- } else {
- tableEnabler.setCaption("Disable table");
- table.setEnabled(true);
- }
- }
-
- public void tableCaption() {
- if (table.getCaption().equals("")) {
- table.setCaption(TABLE_CAPTION);
- tableCaption.setCaption("Hide caption");
- } else {
- table.setCaption("");
- tableCaption.setCaption("Show caption");
- }
- }
-
- /**
- * URIHandler Return example actions
- */
- public Action[] getActions(Object target, Object sender) {
- return actions;
- }
-
- /**
- * Executed by right mouse button on table or tree component.
- */
- public void handleAction(Action action, Object sender, Object target) {
- if (sender == table) {
- tableLastAction.setValue("Last action clicked was '"
- + action.getCaption() + "' on item " + target);
- }
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/ToolkitTunesLayout.java b/src/com/itmill/toolkit/demo/ToolkitTunesLayout.java
index ab8fcdc111..5c8f7774e2 100644
--- a/src/com/itmill/toolkit/demo/ToolkitTunesLayout.java
+++ b/src/com/itmill/toolkit/demo/ToolkitTunesLayout.java
@@ -3,6 +3,7 @@ package com.itmill.toolkit.demo;
import java.util.Random;
import com.itmill.toolkit.Application;
+import com.itmill.toolkit.terminal.ThemeResource;
import com.itmill.toolkit.ui.Button;
import com.itmill.toolkit.ui.ComboBox;
import com.itmill.toolkit.ui.Embedded;
@@ -15,31 +16,30 @@ import com.itmill.toolkit.ui.Table;
import com.itmill.toolkit.ui.VerticalLayout;
import com.itmill.toolkit.ui.Window;
+/**
+ * Sample application layout, similar (almost identical) to Apple iTunes.
+ *
+ * @author IT Mill Ltd.
+ *
+ */
public class ToolkitTunesLayout extends Application {
@Override
public void init() {
- // We'll just build the whole UI here, since it will not contain any
- // logic
+ /*
+ * We'll build the whole UI here, since the application will not contain
+ * any logic. Otherwise it would be more practical to separate parts of
+ * the UI into different classes and methods.
+ */
// Main (browser) window, needed in all Toolkit applications
- // final Window browser = new Window("ToolkitTunes Layout demo");
- // setMainWindow(browser);
-
- // Our player window. We'll make it fill almost the whole browser view,
- // and we'll center it on the screen. Note, that the percentage
- // dimensions in sub-windows only affect the initial render, after the
- // dimensions will be in pixels and will not scale if the user resizes
- // the actual browser window.
final Window root = new Window("ToolkitTunes");
- root.setWidth("90%");
- root.setHeight("90%");
- root.center();
- // We'll attach the window to the browser view already here, so we won't
- // forget it later.
- // browser.addWindow(root);
+ /*
+ * We'll attach the window to the browser view already here, so we won't
+ * forget it later.
+ */
setMainWindow(root);
// Our root window contains one VerticalLayout by default, let's make
@@ -51,7 +51,7 @@ public class ToolkitTunesLayout extends Application {
// modes and search
HorizontalLayout top = new HorizontalLayout();
top.setWidth("100%");
- top.setMargin(true);
+ top.setMargin(false, true, false, true); // Enable horizontal margins
top.setSpacing(true);
// Let's attach that one straight away too
@@ -64,8 +64,7 @@ public class ToolkitTunesLayout extends Application {
HorizontalLayout viewmodes = new HorizontalLayout();
ComboBox search = new ComboBox();
- // Add the components and align them properly (the status component will
- // be the highest of them, so align other components to "middle")
+ // Add the components and align them properly
top.addComponent(playback);
top.addComponent(volume);
top.addComponent(status);
@@ -73,14 +72,17 @@ public class ToolkitTunesLayout extends Application {
top.addComponent(search);
top.setComponentAlignment(playback, "middle");
top.setComponentAlignment(volume, "middle");
+ top.setComponentAlignment(status, "middle");
top.setComponentAlignment(viewmodes, "middle");
top.setComponentAlignment(search, "middle");
- // We want our status area to expand if the user resizes the root
- // window, and we want it to accommodate as much space as there is
- // available. All other components in the top layout should stay fixed
- // sized, so we don't need to specify any expand ratios for them (they
- // will automatically revert to zero after the following line).
+ /*
+ * We want our status area to expand if the user resizes the root
+ * window, and we want it to accommodate as much space as there is
+ * available. All other components in the top layout should stay fixed
+ * sized, so we don't need to specify any expand ratios for them (they
+ * will automatically revert to zero after the following line).
+ */
top.setExpandRatio(status, 1.0F);
// Playback controls
@@ -106,7 +108,7 @@ public class ToolkitTunesLayout extends Application {
// Status area
status.setWidth("80%");
status.setSpacing(true);
- top.setComponentAlignment(status, "center");
+ top.setComponentAlignment(status, "middle center");
Button toggleVisualization = new Button("Mode");
Label timeFromStart = new Label("0:00");
@@ -151,8 +153,10 @@ public class ToolkitTunesLayout extends Application {
viewmodes.addComponent(viewAsGrid);
viewmodes.addComponent(coverflow);
- // That covers the top bar. Now let's move on to the sidebar and track
- // listing
+ /*
+ * That covers the top bar. Now let's move on to the sidebar and track
+ * listing
+ */
// We'll need one splitpanel to separate the sidebar and track listing
SplitPanel bottom = new SplitPanel(SplitPanel.ORIENTATION_HORIZONTAL);
@@ -160,7 +164,6 @@ public class ToolkitTunesLayout extends Application {
// The splitpanel is by default 100% x 100%, but we'll need to adjust
// our main window layout to accomodate the height
- root.getLayout().setHeight("100%");
((VerticalLayout) root.getLayout()).setExpandRatio(bottom, 1.0F);
// Give the sidebar less space than the listing
@@ -172,21 +175,28 @@ public class ToolkitTunesLayout extends Application {
sidebar.setSizeFull();
bottom.setFirstComponent(sidebar);
- // Then we need some labels and buttons, and an album cover image
- // The labels and buttons go into their own vertical layout, since we
- // want the 'sidebar' layout to be expanding (cover image in the
- // bottom).
+ /*
+ * Then we need some labels and buttons, and an album cover image The
+ * labels and buttons go into their own vertical layout, since we want
+ * the 'sidebar' layout to be expanding (cover image in the bottom).
+ * VerticalLayout is by default 100% wide.
+ */
VerticalLayout selections = new VerticalLayout();
Label library = new Label("Library");
Button music = new Button("Music");
+ music.setWidth("100%");
Label store = new Label("Store");
Button toolkitTunesStore = new Button("ToolkitTunes Store");
+ toolkitTunesStore.setWidth("100%");
Button purchased = new Button("Purchased");
+ purchased.setWidth("100%");
Label playlists = new Label("Playlists");
Button genius = new Button("Geniues");
+ genius.setWidth("100%");
Button recent = new Button("Recently Added");
+ recent.setWidth("100%");
// Lets add them to the 'selections' layout
selections.addComponent(library);
@@ -202,20 +212,18 @@ public class ToolkitTunesLayout extends Application {
sidebar.addComponent(selections);
sidebar.setExpandRatio(selections, 1.0F);
- // Then comes the cover artwork
- Embedded cover = new Embedded();
- // We don't have a source image for it yet, but we'll add it later in
- // the theming example
- // cover.setSource(new ThemeResource(""));
- cover.setWidth("200px");
- cover.setHeight("200px");
+ // Then comes the cover artwork (we'll add the actual image in the
+ // themeing section)
+ Embedded cover = new Embedded("Currently Playing");
sidebar.addComponent(cover);
- sidebar.setComponentAlignment(cover, "center");
- // And lastly, we need the track listing table
- // It should fill the whole left side of our bottom layout
+ /*
+ * And lastly, we need the track listing table It should fill the whole
+ * left side of our bottom layout
+ */
Table listing = new Table();
listing.setSizeFull();
+ listing.setSelectable(true);
bottom.setSecondComponent(listing);
// Add the table headers
@@ -264,6 +272,56 @@ public class ToolkitTunesLayout extends Application {
albums[new Random().nextInt(albums.length - 1)],
genres[new Random().nextInt(genres.length - 1)], s }, i);
}
+
+ // We'll align the track time column to right as well
+ listing.setColumnAlignment("Time", Table.ALIGN_RIGHT);
+
+ // TODO the footer
+
+ // Now what's left to do? Themeing of course.
+ setTheme("toolkittunes");
+
+ /*
+ * Let's give a namespace to our application window. This way, if
+ * someone uses the same theme for different applications, we don't get
+ * unwanted style conflicts.
+ */
+ root.setStyleName("tTunes");
+
+ top.setStyleName("top");
+ top.setHeight("75px"); // Same as the background image height
+
+ playback.setStyleName("playback");
+ playback.setMargin(false, true, false, false); // Add right-side margin
+ play.setStyleName("play");
+ next.setStyleName("next");
+ prev.setStyleName("prev");
+ playback.setComponentAlignment(prev, "middle");
+ playback.setComponentAlignment(next, "middle");
+
+ volume.setStyleName("volume");
+ mute.setStyleName("mute");
+ max.setStyleName("max");
+ vol.setWidth("78px");
+
+ status.setStyleName("status");
+ status.setMargin(true);
+ status.setHeight("46px"); // Height of the background image
+
+ toggleVisualization.setStyleName("toggle-vis");
+ jumpToTrack.setStyleName("jump");
+
+ viewAsTable.setStyleName("viewmode-table");
+ viewAsGrid.setStyleName("viewmode-grid");
+ coverflow.setStyleName("viewmode-coverflow");
+
+ sidebar.setStyleName("sidebar");
+
+ music.setStyleName("selected");
+
+ cover.setSource(new ThemeResource("images/album-cover.jpg"));
+ // Because this is an image, it will retain it's aspect ratio
+ cover.setWidth("100%");
}
}
diff --git a/src/com/itmill/toolkit/demo/WindowedDemos.java b/src/com/itmill/toolkit/demo/WindowedDemos.java
deleted file mode 100644
index 26e745bb8e..0000000000
--- a/src/com/itmill/toolkit/demo/WindowedDemos.java
+++ /dev/null
@@ -1,97 +0,0 @@
-/*
-@ITMillApache2LicenseForJavaFiles@
- */
-
-package com.itmill.toolkit.demo;
-
-import java.util.HashMap;
-import java.util.Iterator;
-
-import com.itmill.toolkit.terminal.ExternalResource;
-import com.itmill.toolkit.ui.Button;
-import com.itmill.toolkit.ui.Embedded;
-import com.itmill.toolkit.ui.Window;
-import com.itmill.toolkit.ui.Button.ClickEvent;
-
-/**
- * Embeds other demos in windows using an ExternalResource ("application in
- * application").
- *
- * @author IT Mill Ltd.
- * @see com.itmill.toolkit.ui.Window
- */
-public class WindowedDemos extends com.itmill.toolkit.Application {
-
- // keeps track of created windows
- private final HashMap windows = new HashMap();
-
- // mapping demo name to URL
- private static final HashMap servlets = new HashMap();
- static {
- servlets.put("Caching demo", "CachingDemo/");
- servlets.put("Calculator", "Calc/");
- servlets.put("Calendar demo", "CalendarDemo/");
- servlets.put("Select demo", "SelectDemo/");
- servlets.put("Table demo", "TableDemo/");
- servlets.put("Browser demo", "BrowserDemo/");
- servlets.put("Notification demo", "NotificationDemo/");
- }
-
- @Override
- public void init() {
-
- // Create new window for the application and give the window a visible.
- final Window main = new Window("IT Mill Toolkit 5 Windowed Demos");
- // set as main window
- setMainWindow(main);
-
- // Create menu window.
- final Window menu = new Window("Select demo");
- menu.setWidth(200);
- menu.setHeight(400);
- main.addWindow(menu); // add to layout
-
- // Create a menu button for each demo
- for (final Iterator it = servlets.keySet().iterator(); it.hasNext();) {
- final String name = (String) it.next();
- final Button b = new Button(name, new Button.ClickListener() {
- public void buttonClick(ClickEvent event) {
- show(event.getButton().getCaption());
- }
-
- });
- b.setStyleName("link");
- menu.addComponent(b);
- }
-
- }
-
- /**
- * Shows the specified demo in a separate window. Creates a new window if
- * the demo has not been shown already, re-uses old window otherwise.
- *
- * @param demoName
- * the name of the demo to be shown
- */
- private void show(String demoName) {
- Window w = (Window) windows.get(demoName);
- if (w == null) {
- w = new Window(demoName);
- w.setWidth(560);
- w.setHeight(500);
- w.setPositionX(202);
- w.getLayout().setSizeFull();
- w.getLayout().setMargin(false);
- windows.put(demoName, w);
- Embedded emb = new Embedded();
- emb.setType(Embedded.TYPE_BROWSER);
- emb
- .setSource(new ExternalResource((String) servlets
- .get(demoName)));
- emb.setSizeFull();
- w.addComponent(emb);
- }
- getMainWindow().addWindow(w);
- }
-
-}
diff --git a/src/com/itmill/toolkit/demo/coverflow/Coverflow.java b/src/com/itmill/toolkit/demo/coverflow/Coverflow.java
new file mode 100644
index 0000000000..a12aa7021c
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/Coverflow.java
@@ -0,0 +1,89 @@
+/*
+@ITMillApache2LicenseForJavaFiles@
+ */
+
+package com.itmill.toolkit.demo.coverflow;
+
+import com.itmill.toolkit.terminal.PaintException;
+import com.itmill.toolkit.terminal.PaintTarget;
+import com.itmill.toolkit.ui.AbstractSelect;
+
+public class Coverflow extends AbstractSelect {
+
+ private String backgroundGradientStart = "FFFFFF";
+ private String backgroundGradientEnd = "EEEEEE";
+ private boolean scrollbarVisibility = true;
+
+ public String getTag() {
+ return "cover";
+ }
+
+ /**
+ * Paints the uidl
+ * @param PaintTarget target
+ * @throws PaintException
+ */
+ public void paintContent(PaintTarget target) throws PaintException {
+ // Superclass writes any common attributes in the paint target.
+ super.paintContent(target);
+
+ target.addAttribute("backgroundGradientStart", backgroundGradientStart);
+ target.addAttribute("backgroundGradientEnd", backgroundGradientEnd);
+ target.addAttribute("scrollbarVisibility", scrollbarVisibility);
+ }
+
+ /**
+ * The user can specify a background gradient for the coverflow. The input values
+ * are RGB values for the start and end gradients.
+ * @param int startR
+ * @param int startG
+ * @param int startB
+ * @param int endR
+ * @param int endG
+ * @param int endB
+ */
+ public void setBackgroundColor(int startR, int startG, int startB, int endR, int endG, int endB) {
+ backgroundGradientStart = "";
+ backgroundGradientEnd = "";
+
+ // Convert all integers to hexadecimal format and make sure they are two characters long (in
+ // other words, add a zero infront if the value is less than 16 => 0x0F
+ if(startR < 16)
+ backgroundGradientStart += "0";
+ backgroundGradientStart += Integer.toHexString(Math.max(Math.min(startR,255),0));
+
+ if(startG < 16)
+ backgroundGradientStart += "0";
+ backgroundGradientStart += Integer.toHexString(Math.max(Math.min(startG,255),0));
+
+ if(startB < 16)
+ backgroundGradientStart += "0";
+ backgroundGradientStart += Integer.toHexString(Math.max(Math.min(startB,255),0));
+
+ if(endR < 16)
+ backgroundGradientEnd += "0";
+ backgroundGradientEnd += Integer.toHexString(Math.max(Math.min(endR,255),0));
+
+ if(endG < 16)
+ backgroundGradientEnd += "0";
+ backgroundGradientEnd += Integer.toHexString(Math.max(Math.min(endG,255),0));
+
+ if(endB < 16)
+ backgroundGradientEnd += "0";
+ backgroundGradientEnd += Integer.toHexString(Math.max(Math.min(endB,255),0));
+
+ this.requestRepaint();
+ }
+
+ /**
+ * The user can toggle the visibility of the scrollbar
+ * @param boolean visible
+ */
+ public void setScrollbarVisibility(boolean visible) {
+ if(scrollbarVisibility != visible) {
+ scrollbarVisibility = visible;
+ this.requestRepaint();
+ }
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.html b/src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.html
new file mode 100644
index 0000000000..59f87c2123
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.html
@@ -0,0 +1,149 @@
+<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
+<html><head>
+<title></title>
+ <style type="text/css">
+ <!--code { font-family: Courier New, Courier; font-size: 10pt; margin: 0px; }-->
+ </style>
+ <meta http-equiv="Content-Type" content="text/html; charset=iso-8859-1" />
+</head><body>
+
+
+<!-- ======================================================== -->
+<!-- = Java Sourcecode to HTML automatically converted code = -->
+<!-- = Java2Html Converter 5.0 [2006-02-26] by Markus Gebhard markus@jave.de = -->
+<!-- = Further information: http://www.java2html.de = -->
+<div align="left" class="java">
+<table border="0" cellpadding="3" cellspacing="0" bgcolor="#ffffff">
+ <tr>
+ <!-- start source code -->
+ <td nowrap="nowrap" valign="top" align="left">
+ <code>
+<font color="#7f0055"><b>package&nbsp;</b></font><font color="#000000">com.itmill.toolkit.demo.coverflow;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.BufferedReader;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.FileInputStream;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.FileReader;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.IOException;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.InputStream;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.InputStreamReader;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.Reader;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.io.StringReader;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.net.URL;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">java.nio.CharBuffer;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.data.Property;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.terminal.Resource;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.terminal.Sizeable;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.terminal.ThemeResource;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.Button;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.Embedded;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.ExpandLayout;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.Label;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.Window;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.Button.ClickEvent;</font><br />
+<font color="#7f0055"><b>import&nbsp;</b></font><font color="#000000">com.itmill.toolkit.ui.Layout.AlignmentHandler;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#7f0055"><b>public&nbsp;class&nbsp;</b></font><font color="#000000">CoverflowApplication&nbsp;</font><font color="#7f0055"><b>extends&nbsp;</b></font><font color="#000000">com.itmill.toolkit.Application&nbsp;</font><font color="#000000">{</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#000000">Coverflow&nbsp;covers&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Coverflow</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#7f0055"><b>public&nbsp;</b></font><font color="#7f0055"><b>void&nbsp;</b></font><font color="#000000">init</font><font color="#000000">()&nbsp;{</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">setMainWindow</font><font color="#000000">(</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Window</font><font color="#000000">(</font><font color="#2a00ff">&#34;Coverflow&#34;</font><font color="#000000">,&nbsp;createMainLayout</font><font color="#000000">()))</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">setTheme</font><font color="#000000">(</font><font color="#2a00ff">&#34;black&#34;</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">addSlidesToCoverflow</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#7f0055"><b>private&nbsp;</b></font><font color="#000000">ExpandLayout&nbsp;createMainLayout</font><font color="#000000">()&nbsp;{</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#3f7f5f">//&nbsp;Initialize&nbsp;coverflow&nbsp;component</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">covers.setHeight</font><font color="#000000">(</font><font color="#990000">150</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">covers.setWidth</font><font color="#000000">(</font><font color="#990000">100</font><font color="#000000">,&nbsp;Sizeable.UNITS_PERCENTAGE</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">covers.setBackgroundColor</font><font color="#000000">(</font><font color="#990000">0</font><font color="#000000">,&nbsp;</font><font color="#990000">0</font><font color="#000000">,&nbsp;</font><font color="#990000">0</font><font color="#000000">,&nbsp;</font><font color="#990000">100</font><font color="#000000">,&nbsp;</font><font color="#990000">100</font><font color="#000000">,&nbsp;</font><font color="#990000">100</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#3f7f5f">//&nbsp;Initialize&nbsp;visible&nbsp;slide&nbsp;viewer</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>final&nbsp;</b></font><font color="#000000">Embedded&nbsp;visibleSlide&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Embedded</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">visibleSlide.setHeight</font><font color="#000000">(</font><font color="#990000">100</font><font color="#000000">,&nbsp;Sizeable.UNITS_PERCENTAGE</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#3f7f5f">//&nbsp;Listen&nbsp;to&nbsp;coverflow&nbsp;changes&nbsp;as&nbsp;change&nbsp;slides&nbsp;when&nbsp;needed</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">covers.addListener</font><font color="#000000">(</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Property.ValueChangeListener</font><font color="#000000">()&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>public&nbsp;</b></font><font color="#7f0055"><b>void&nbsp;</b></font><font color="#000000">valueChange</font><font color="#000000">(</font><font color="#000000">Property.ValueChangeEvent&nbsp;event</font><font color="#000000">)&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">visibleSlide.setSource</font><font color="#000000">((</font><font color="#000000">Resource</font><font color="#000000">)&nbsp;</font><font color="#000000">covers.getValue</font><font color="#000000">())</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">})</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#3f7f5f">//&nbsp;Show&nbsp;sources&nbsp;button</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">Button&nbsp;showSrc&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Button</font><font color="#000000">(</font><font color="#2a00ff">&#34;src&#34;</font><font color="#000000">,&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Button.ClickListener</font><font color="#000000">()&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>public&nbsp;</b></font><font color="#7f0055"><b>void&nbsp;</b></font><font color="#000000">buttonClick</font><font color="#000000">(</font><font color="#000000">ClickEvent&nbsp;event</font><font color="#000000">)&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">Window&nbsp;srcWindow&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Window</font><font color="#000000">(</font><font color="#2a00ff">&#34;Source&nbsp;code&#34;</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">srcWindow.setWidth</font><font color="#000000">(</font><font color="#990000">700</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">srcWindow.setHeight</font><font color="#000000">(</font><font color="#990000">500</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">Label&nbsp;l&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">Label</font><font color="#000000">(</font><font color="#000000">getSourceCodeForThisClass</font><font color="#000000">()</font><font color="#000000">,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">Label.CONTENT_XHTML</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">srcWindow.addComponent</font><font color="#000000">(</font><font color="#000000">l</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">getMainWindow</font><font color="#000000">()</font><font color="#000000">.addWindow</font><font color="#000000">(</font><font color="#000000">srcWindow</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">})</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#3f7f5f">//&nbsp;Initialize&nbsp;main&nbsp;layout</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">ExpandLayout&nbsp;layout&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">ExpandLayout</font><font color="#000000">(</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">ExpandLayout.ORIENTATION_VERTICAL</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.addComponent</font><font color="#000000">(</font><font color="#000000">showSrc</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.setComponentAlignment</font><font color="#000000">(</font><font color="#000000">showSrc,&nbsp;AlignmentHandler.ALIGNMENT_RIGHT,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">AlignmentHandler.ALIGNMENT_TOP</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.addComponent</font><font color="#000000">(</font><font color="#000000">visibleSlide</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.setComponentAlignment</font><font color="#000000">(</font><font color="#000000">visibleSlide,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">AlignmentHandler.ALIGNMENT_HORIZONTAL_CENTER,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">AlignmentHandler.ALIGNMENT_TOP</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.addComponent</font><font color="#000000">(</font><font color="#000000">covers</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.setComponentAlignment</font><font color="#000000">(</font><font color="#000000">covers,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">AlignmentHandler.ALIGNMENT_HORIZONTAL_CENTER,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">AlignmentHandler.ALIGNMENT_TOP</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.expand</font><font color="#000000">(</font><font color="#000000">visibleSlide</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">layout.setSizeFull</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>return&nbsp;</b></font><font color="#000000">layout;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#7f0055"><b>private&nbsp;</b></font><font color="#000000">String&nbsp;getSourceCodeForThisClass</font><font color="#000000">()&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">String&nbsp;code&nbsp;=&nbsp;</font><font color="#2a00ff">&#34;Could&nbsp;not&nbsp;find&nbsp;source-file&#34;</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>try&nbsp;</b></font><font color="#000000">{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">InputStream&nbsp;is&nbsp;=&nbsp;</font><font color="#7f0055"><b>this</b></font><font color="#000000">.getClass</font><font color="#000000">()</font><font color="#000000">.getResource</font><font color="#000000">(</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#2a00ff">&#34;CoverflowApplication.html&#34;</font><font color="#000000">)</font><font color="#000000">.openStream</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">BufferedReader&nbsp;r&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">BufferedReader</font><font color="#000000">(</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">InputStreamReader</font><font color="#000000">(</font><font color="#000000">is</font><font color="#000000">))</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">StringBuffer&nbsp;buf&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">StringBuffer</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">String&nbsp;line;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>while&nbsp;</b></font><font color="#000000">((</font><font color="#000000">line&nbsp;=&nbsp;r.readLine</font><font color="#000000">())&nbsp;</font><font color="#000000">!=&nbsp;</font><font color="#7f0055"><b>null</b></font><font color="#000000">)&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">buf.append</font><font color="#000000">(</font><font color="#000000">line</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">code&nbsp;=&nbsp;buf.toString</font><font color="#000000">()</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">}&nbsp;</font><font color="#7f0055"><b>catch&nbsp;</b></font><font color="#000000">(</font><font color="#000000">IOException&nbsp;ignored</font><font color="#000000">)&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>return&nbsp;</b></font><font color="#000000">code;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff"></font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#7f0055"><b>private&nbsp;</b></font><font color="#7f0055"><b>void&nbsp;</b></font><font color="#000000">addSlidesToCoverflow</font><font color="#000000">()&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>for&nbsp;</b></font><font color="#000000">(</font><font color="#7f0055"><b>int&nbsp;</b></font><font color="#000000">i&nbsp;=&nbsp;</font><font color="#990000">1</font><font color="#000000">;&nbsp;i&nbsp;&lt;=&nbsp;</font><font color="#990000">22</font><font color="#000000">;&nbsp;i++</font><font color="#000000">)&nbsp;{</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">String&nbsp;head&nbsp;=&nbsp;</font><font color="#2a00ff">&#34;../../../IMAGES/&#34;</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">String&nbsp;tail&nbsp;=&nbsp;</font><font color="#2a00ff">&#34;slideshow-example.0&#34;&nbsp;</font><font color="#000000">+&nbsp;</font><font color="#000000">((</font><font color="#000000">i&nbsp;&lt;&nbsp;</font><font color="#990000">10</font><font color="#000000">)&nbsp;</font><font color="#000000">?&nbsp;</font><font color="#2a00ff">&#34;0&#34;&nbsp;</font><font color="#000000">:&nbsp;</font><font color="#2a00ff">&#34;&#34;</font><font color="#000000">)&nbsp;</font><font color="#000000">+&nbsp;i</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">+&nbsp;</font><font color="#2a00ff">&#34;.jpg&#34;</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">ThemeResource&nbsp;slide&nbsp;=&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">ThemeResource</font><font color="#000000">(</font><font color="#000000">head&nbsp;+&nbsp;tail</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">covers.addItem</font><font color="#000000">(</font><font color="#000000">slide</font><font color="#000000">)</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">covers.setItemIcon</font><font color="#000000">(</font><font color="#000000">slide,</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#7f0055"><b>new&nbsp;</b></font><font color="#000000">ThemeResource</font><font color="#000000">(</font><font color="#000000">head&nbsp;+&nbsp;</font><font color="#2a00ff">&#34;thumbs/&#34;&nbsp;</font><font color="#000000">+&nbsp;tail</font><font color="#000000">))</font><font color="#000000">;</font><br />
+<font color="#ffffff">&nbsp;&nbsp;&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#ffffff">&nbsp;&nbsp;</font><font color="#000000">}</font><br />
+<font color="#000000">}</font></code>
+
+ </td>
+ <!-- end source code -->
+ </tr>
+</table>
+</div>
+<!-- = END of automatically generated HTML code = -->
+<!-- ======================================================== -->
+
+
+</body></html> \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.java b/src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.java
new file mode 100644
index 0000000000..3fa4cd3155
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/CoverflowApplication.java
@@ -0,0 +1,109 @@
+/*
+@ITMillApache2LicenseForJavaFiles@
+ */
+
+package com.itmill.toolkit.demo.coverflow;
+
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+
+import com.itmill.toolkit.data.Property;
+import com.itmill.toolkit.terminal.Resource;
+import com.itmill.toolkit.terminal.Sizeable;
+import com.itmill.toolkit.terminal.ThemeResource;
+import com.itmill.toolkit.ui.Alignment;
+import com.itmill.toolkit.ui.Button;
+import com.itmill.toolkit.ui.Embedded;
+import com.itmill.toolkit.ui.Label;
+import com.itmill.toolkit.ui.VerticalLayout;
+import com.itmill.toolkit.ui.Window;
+import com.itmill.toolkit.ui.Button.ClickEvent;
+
+public class CoverflowApplication extends com.itmill.toolkit.Application {
+
+ Coverflow covers = new Coverflow();
+
+ public void init() {
+
+ setMainWindow(new Window("Coverflow", createMainLayout()));
+
+ setTheme("coverflow");
+
+ addSlidesToCoverflow();
+ }
+
+ private VerticalLayout createMainLayout() {
+
+ // Initialize coverflow component
+ covers.setHeight("150px");
+ covers.setWidth("100%");
+ covers.setBackgroundColor(0, 0, 0, 100, 100, 100);
+
+ // Initialize visible slide viewer
+ final Embedded visibleSlide = new Embedded();
+ visibleSlide.setHeight(100, Sizeable.UNITS_PERCENTAGE);
+
+ // Listen to coverflow changes as change slides when needed
+ covers.addListener(new Property.ValueChangeListener() {
+ public void valueChange(Property.ValueChangeEvent event) {
+ visibleSlide.setSource((Resource) covers.getValue());
+ }
+ });
+
+ // Show sources button
+ Button showSrc = new Button("src", new Button.ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ Window srcWindow = new Window("Source code");
+ srcWindow.setWidth("700px");
+ srcWindow.setHeight("500px");
+ Label l = new Label(getSourceCodeForThisClass(),
+ Label.CONTENT_XHTML);
+ srcWindow.addComponent(l);
+ getMainWindow().addWindow(srcWindow);
+ }
+ });
+
+ // Initialize main layout
+ VerticalLayout layout = new VerticalLayout();
+ layout.addComponent(showSrc);
+ layout.setComponentAlignment(showSrc, Alignment.TOP_RIGHT);
+ layout.addComponent(visibleSlide);
+ layout.setComponentAlignment(visibleSlide, Alignment.TOP_CENTER);
+ layout.addComponent(covers);
+ layout.setExpandRatio(visibleSlide, 1);
+ layout.setSizeFull();
+
+ return layout;
+ }
+
+ private String getSourceCodeForThisClass() {
+ String code = "Could not find source-file";
+ try {
+ InputStream is = this.getClass().getResource(
+ "CoverflowApplication.html").openStream();
+ BufferedReader r = new BufferedReader(new InputStreamReader(is));
+ StringBuffer buf = new StringBuffer();
+ String line;
+ while ((line = r.readLine()) != null) {
+ buf.append(line);
+ }
+ code = buf.toString();
+ } catch (IOException ignored) {
+ }
+ return code;
+ }
+
+ private void addSlidesToCoverflow() {
+ for (int i = 1; i <= 22; i++) {
+ String head = "images/";
+ String tail = "slideshow-example.0" + ((i < 10) ? "0" : "") + i
+ + ".jpg";
+ ThemeResource slide = new ThemeResource(head + tail);
+ covers.addItem(slide);
+ covers.setItemIcon(slide,
+ new ThemeResource(head + "thumbs/" + tail));
+ }
+ }
+}
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/CoverflowWidgetSet.gwt.xml b/src/com/itmill/toolkit/demo/coverflow/gwt/CoverflowWidgetSet.gwt.xml
new file mode 100644
index 0000000000..a3a4e1d2ce
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/CoverflowWidgetSet.gwt.xml
@@ -0,0 +1,8 @@
+<module>
+ <!-- Inherit the NoEntry version to avoid multiple entrypoints -->
+ <inherits name="com.itmill.toolkit.terminal.gwt.DefaultWidgetSetNoEntry" />
+
+ <!-- Entry point -->
+ <entry-point class="com.itmill.toolkit.demo.coverflow.gwt.client.CoverflowWidgetSet"/>
+
+</module>
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/client/CoverflowWidgetSet.java b/src/com/itmill/toolkit/demo/coverflow/gwt/client/CoverflowWidgetSet.java
new file mode 100644
index 0000000000..3d353d7809
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/client/CoverflowWidgetSet.java
@@ -0,0 +1,35 @@
+/*
+@ITMillApache2LicenseForJavaFiles@
+ */
+
+package com.itmill.toolkit.demo.coverflow.gwt.client;
+
+import com.itmill.toolkit.demo.coverflow.gwt.client.ui.ICoverflow;
+import com.itmill.toolkit.terminal.gwt.client.DefaultWidgetSet;
+import com.itmill.toolkit.terminal.gwt.client.Paintable;
+import com.itmill.toolkit.terminal.gwt.client.UIDL;
+
+public class CoverflowWidgetSet extends DefaultWidgetSet {
+ /** Creates a widget according to its class name. */
+ public Paintable createWidget(UIDL uidl) {
+ final String className = resolveWidgetTypeName(uidl);
+ if ("com.itmill.toolkit.demo.coverflow.gwt.client.ui.ICoverflow"
+ .equals(className)) {
+ return new ICoverflow();
+ }
+
+ // Let the DefaultWidgetSet handle creation of default widgets
+ return super.createWidget(uidl);
+ }
+
+ /** Resolves UIDL tag name to class name. */
+ protected String resolveWidgetTypeName(UIDL uidl) {
+ final String tag = uidl.getTag();
+ if ("cover".equals(tag)) {
+ return "com.itmill.toolkit.demo.coverflow.gwt.client.ui.ICoverflow";
+ }
+
+ // Let the DefaultWidgetSet handle resolution of default widgets
+ return super.resolveWidgetTypeName(uidl);
+ }
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/client/ui/ICoverflow.java b/src/com/itmill/toolkit/demo/coverflow/gwt/client/ui/ICoverflow.java
new file mode 100644
index 0000000000..4a452f2ba2
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/client/ui/ICoverflow.java
@@ -0,0 +1,336 @@
+/*
+@ITMillApache2LicenseForJavaFiles@
+ */
+
+package com.itmill.toolkit.demo.coverflow.gwt.client.ui;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Iterator;
+
+import com.google.gwt.core.client.GWT;
+import com.google.gwt.user.client.ui.Composite;
+import com.google.gwt.user.client.ui.HTML;
+import com.itmill.toolkit.terminal.gwt.client.ApplicationConnection;
+import com.itmill.toolkit.terminal.gwt.client.Paintable;
+import com.itmill.toolkit.terminal.gwt.client.UIDL;
+
+public class ICoverflow extends Composite implements Paintable {
+ private String uidlId;
+ protected ApplicationConnection client;
+ private ArrayList coverList = new ArrayList();
+
+ private Object _selected;
+ private boolean flashInited = false;
+ private HTML flash;
+ private boolean scrollbarVisibility = true;
+ private String backgroundGradientStart;
+ private String backgroundGradientEnd;
+ private boolean colorChanged = false;
+ private boolean sbVisibilityChanged = false;
+ private HashMap keyMap = new HashMap();
+
+ /**
+ * Constructor
+ */
+ public ICoverflow() {
+ flash = new HTML();
+
+ initWidget(flash);
+ }
+
+ /**
+ * This method accepts parses the uidl sent by the server
+ *
+ * @param UIDL
+ * uidl
+ * @param ApplicationConnection
+ * client
+ */
+ public void updateFromUIDL(UIDL uidl, ApplicationConnection client) {
+ // Store variables
+ uidlId = uidl.getId();
+ this.client = client;
+ String tempColor;
+
+ if (client.updateComponent(this, uidl, true)) {
+ return;
+ }
+
+ // Has the scrollbar's visibility status changed?
+ if (uidl.hasAttribute("scrollbarVisibility")) {
+ boolean tempVis = uidl.getBooleanAttribute("scrollbarVisibility");
+ if (scrollbarVisibility != tempVis) {
+ scrollbarVisibility = tempVis;
+ sbVisibilityChanged = true;
+ }
+ }
+
+ // Has the start gradient changed?
+ if (uidl.hasAttribute("backgroundGradientStart")) {
+ tempColor = uidl.getStringAttribute("backgroundGradientStart")
+ .toString();
+ if (tempColor != backgroundGradientStart) {
+ backgroundGradientStart = tempColor;
+ colorChanged = true;
+ }
+ }
+
+ // Has the end gradient changed?
+ if (uidl.hasAttribute("backgroundGradientEnd")) {
+ tempColor = uidl.getStringAttribute("backgroundGradientEnd")
+ .toString();
+ if (tempColor != backgroundGradientEnd) {
+ backgroundGradientEnd = tempColor;
+ colorChanged = true;
+ }
+ }
+
+ final UIDL images = uidl.getChildUIDL(0);
+
+ // Check which covers should be removed. This array list contains all
+ // current
+ // covers. We remove from this list all covers which are sent with the
+ // repainted
+ // uidl. All remaining covers in this list should be "old" ones and are
+ // should
+ // be deleted.
+
+ ArrayList newList = new ArrayList();
+
+ // Iterate through all option elements
+ for (final Iterator i = images.getChildIterator(); i.hasNext();) {
+ final UIDL imgUidl = (UIDL) i.next();
+
+ // Make sure all required attributes exist
+ if (imgUidl.hasAttribute("caption") && imgUidl.hasAttribute("key")
+ && imgUidl.hasAttribute("icon")) {
+ HashMap set = new HashMap();
+
+ // Update the key map
+ keyMap.put(imgUidl.getStringAttribute("caption"), imgUidl
+ .getStringAttribute("key"));
+
+ // Get information
+
+ set.put("icon", client.translateToolkitUri(imgUidl
+ .getStringAttribute("icon")));
+ set.put("caption", imgUidl.getStringAttribute("caption"));
+
+ newList.add(set);
+
+ // Is the current cover selected?
+ if (imgUidl.hasAttribute("selected")) {
+ _selected = imgUidl.getStringAttribute("caption");
+ }
+ }
+ }
+
+ // Deleted items
+ ArrayList intersectList = new ArrayList();
+ intersectList.addAll(coverList);
+ intersectList.removeAll(newList);
+
+ if (flashInited) {
+ for (int i = 0; i < intersectList.size(); i++) {
+ HashMap cover = (HashMap) intersectList.get(i);
+ removeCover(uidlId, cover.get("caption").toString());
+ }
+ }
+
+ // Added items
+ intersectList = new ArrayList();
+ intersectList.addAll(newList);
+ intersectList.removeAll(coverList);
+
+ if (flashInited) {
+ for (int i = 0; i < intersectList.size(); i++) {
+ HashMap cover = (HashMap) intersectList.get(i);
+ addCover(uidlId, cover.get("caption").toString(), cover.get(
+ "icon").toString());
+ }
+ }
+
+ coverList = newList;
+
+ // Has the flash been initialized?
+ if (!flashInited) {
+ colorChanged = false;
+ setFlash();
+ initializeMethods(uidlId);
+ }
+
+ // Inform flash of the selected cover
+ if (_selected != null && flashInited) {
+ selectCover(uidlId, _selected.toString());
+ }
+
+ if (colorChanged && flashInited) {
+ setBackgroundColor(uidlId, backgroundGradientStart,
+ backgroundGradientEnd);
+ colorChanged = false;
+ }
+
+ if (sbVisibilityChanged && flashInited) {
+ toggleScrollbarVisibility(uidlId, scrollbarVisibility);
+ sbVisibilityChanged = false;
+ }
+
+ }
+
+ /**
+ * Inform the server which cover is selected
+ *
+ * @param String
+ * coverKey
+ */
+ public void setCover(String coverId) {
+ if (uidlId == null || client == null) {
+ return;
+ }
+
+ client.updateVariable(uidlId, "selected", new String[] { keyMap.get(
+ coverId).toString() }, true);
+ }
+
+ /**
+ * Initialize the native javascript functions needed for the flash <-> GWT
+ * communication
+ *
+ * @param String
+ * id
+ */
+ public native void initializeMethods(String id) /*-{
+ var app = this;
+
+ if($wnd.itmill.coverflow == null)
+ var coverflow = [];
+ else
+ var coverflow = $wnd.itmill.coverflow;
+
+ coverflow['getCovers_' + id] = function() {
+ app.@com.itmill.toolkit.demo.coverflow.gwt.client.ui.ICoverflow::getCovers()();
+ };
+
+ coverflow['setCurrent_' + id] = function(selected) {
+ app.@com.itmill.toolkit.demo.coverflow.gwt.client.ui.ICoverflow::setCover(Ljava/lang/String;)(selected);
+ };
+
+ $wnd.itmill.coverflow = coverflow;
+ }-*/;
+
+ /**
+ * This function sends all covers to the flash. We cannot do this directly
+ * in the updateFromUIDL method, because we cannot be sure if the flash has
+ * been loaded into the browser. The flash will call for this method when
+ * it's ready.
+ */
+ public void getCovers() {
+ // Loop through all stored coves
+ for (int i = 0; i < coverList.size(); i++) {
+ HashMap set = (HashMap) coverList.get(i);
+
+ try {
+ // Add the cover
+ addCover(uidlId, set.get("caption").toString(), set.get("icon")
+ .toString());
+ } catch (Exception e) {
+ // Do not add covers lacking obligatory data
+ }
+ }
+ // The flash calls for this method, therefore we can be sure that the
+ // flash has been loaded
+ // into the browser.
+ flashInited = true;
+
+ // Set selected cover
+ if (_selected != null) {
+ selectCover(uidlId, _selected.toString());
+ }
+ }
+
+ /**
+ * This function is a native javascript function which adds covers to the
+ * actual flash. This method works as a bridge between GWT and flash.
+ *
+ * @param id
+ * @param key
+ * @param caption
+ * @param icon
+ */
+ public native void addCover(String id, String caption, String icon) /*-{
+ try {
+ $doc['fxcoverflow' + id].addCover(caption.toString(), icon.toString());
+ }
+ catch(e) {
+ $wnd.alert(e.message);
+ }
+
+ }-*/;
+
+ /**
+ * This function tells the flash which cover should be selected.
+ *
+ * @param id
+ * @param key
+ */
+ public native void selectCover(String id, String key) /*-{
+ $doc["fxcoverflow" + id].selectCover(key.toString());
+ }-*/;
+
+ public native void setBackgroundColor(String id, String startGradient,
+ String endGradient) /*-{
+ $doc["fxcoverflow" + id].setBackgroundColor("0x" + startGradient.toString(), "0x" + endGradient.toString());
+ }-*/;
+
+ public native void toggleScrollbarVisibility(String id, boolean visibility) /*-{
+ $doc["fxcoverflow" + id].toggleScrollbarVisibility(visibility);
+ }-*/;
+
+ public native void removeCover(String id, String key) /*-{
+ $doc["fxcoverflow" + id].removeCover(key);
+ }-*/;
+
+ /**
+ * Set the HTML coding of the flash movie. This isn't done until the
+ * updateFromUIDL method is called for the first time. The reason is that we
+ * use an id from the UIDL to uniquely identify all instances of this
+ * widget.
+ */
+ private void setFlash() {
+ String html = "<object id=\"fxcoverflow"
+ + uidlId
+ + "\" classid=\"clsid:d27cdb6e-ae6d-11cf-96b8-444553540000\" width=\"100%\""
+ + " height=\"100%\" codebase=\"http://fpdownload.adobe.com/pub/shockwave/cabs/flash/swflash.cab#version=9,0,0,0\">"
+ + "<param name=\"movie\" value=\""
+ + GWT.getModuleBaseURL()
+ + "coverflowflash.swf\">"
+ + "<param name=\"quality\" value=\"high\">"
+ + "<param name=\"flashVars\" value=\"pid="
+ + uidlId
+ + "&sbVis="
+ + scrollbarVisibility
+ + "&bgS=0x"
+ + backgroundGradientStart
+ + "&bgE=0x"
+ + backgroundGradientEnd
+ + "\" />"
+ + "<embed name=\"fxcoverflow"
+ + uidlId
+ + "\" flashVars=\"pid="
+ + uidlId
+ + "&sbVis="
+ + scrollbarVisibility
+ + "&bgS=0x"
+ + backgroundGradientStart
+ + "&bgE=0x"
+ + backgroundGradientEnd
+ + "\" src=\""
+ + GWT.getModuleBaseURL()
+ + "coverflowflash.swf\" width=\"100%\" height=\"100%\" "
+ + "quality=\"high\" "
+ + "pluginspage=\"http://www.adobe.com/shockwave/download/index.cgi?P1_Prod_Version=ShockwaveFlash\">"
+ + "</embed>" + "</object>";
+ flash.setHTML(html);
+ }
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/Cover.as b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/Cover.as
new file mode 100644
index 0000000000..61b08ead48
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/Cover.as
@@ -0,0 +1,392 @@
+package {
+ import flash.events.Event;
+ import flash.display.Loader;
+ import flash.display.DisplayObject;
+ import flash.system.LoaderContext;
+ import flash.display.Bitmap;
+ import flash.display.BitmapData;
+ import flash.net.URLRequest;
+ import mx.controls.Alert;
+
+ import flash.system.Security;
+ import flash.system.ApplicationDomain;
+
+ import mx.controls.Image;
+ import mx.core.UIComponent;
+ import flash.geom.Matrix;
+ import flash.display.Sprite;
+ import flash.display.Shape;
+ import flash.display.Graphics;
+ import flash.display.GradientType;
+ import flash.geom.Rectangle;
+ import flash.geom.Point;
+
+ import sandy.util.*;
+
+ public class Cover extends UIComponent {
+ // Cover information
+ private var _caption:String;
+ private var _uri:String;
+ private var _imageLoaded:Boolean = false;
+
+ // Actual content information
+ private var _bitmapData:BitmapData;
+ private var _bitmap:Bitmap;
+ private var _img:Image;
+ private var _distortedShape:Shape;
+ private var _reflectionBitmap:Bitmap;
+ private var _reflectionShape:Shape;
+
+
+ // Distort information
+ private var _realAngle:Number = -1;
+ private var _realScale:Number = -1;
+ private var _angle:Number = 0;
+ private var _scale:Number = 1;
+ private static const perspectiveConstant:Number = .15;
+
+
+ // Position information
+ private var _x:int;
+ private var _y:int;
+
+ /**
+ * Constructor
+ */
+ public function Cover(caption:String, uri:String) {
+ super();
+
+ // Set this element's size to 100% x 100%
+ super.percentHeight = 100;
+ super.percentWidth = 100;
+
+ // Store input data
+ this._caption = caption;
+ this._uri = uri;
+
+ // Initialize default image
+ this._img = new Image();
+ this._img.percentHeight = 100;
+ this._img.percentWidth = 100;
+ this._bitmapData = new BitmapData(100,100,false, 0xffff0000);
+
+ _distortedShape = new Shape();
+ addChild(_distortedShape);
+
+ _reflectionShape = new Shape();
+ addChild(_reflectionShape);
+
+ // Create the default image
+ this._img.source = this.getBitmap();
+
+ addChildAt(_img, 0);
+ }
+
+ /**
+ * Load an image if it hasn't been loaded yet
+ */
+ public function loadImage():void {
+ if(!_imageLoaded) {
+ // Create a loader to load the image
+ var request:URLRequest = new URLRequest(this._uri);
+ var imageLoader:Loader = new Loader();
+
+ // Check for restrictions
+ var imgLdrContext:LoaderContext = new LoaderContext(false, ApplicationDomain.currentDomain);
+ imgLdrContext.checkPolicyFile = true;
+
+ // Set an event listener
+ imageLoader.contentLoaderInfo.addEventListener(Event.COMPLETE, imgLoaded);
+
+ // Load the image
+ imageLoader.load(request, imgLdrContext);
+ }
+ }
+
+ /**
+ * Event handler for image load completion
+ * @param Event event
+ */
+ private function imgLoaded(event:Event):void {
+ // Check if we got the image
+ try {
+ removeChild(_img);
+ this._bitmapData = Bitmap(event.currentTarget.content).bitmapData;
+
+ // Set new source for the image
+ this._img.source = this.getBitmap();
+ addChildAt(_img, 0);
+
+ _imageLoaded = true;
+ invalidateDisplayList();
+ }
+ catch(error:Error) {
+ // An error occured
+ Alert.show(error.toString());
+ }
+ }
+
+ /**
+ * Measure the size of the image
+ */
+ override protected function measure():void {
+ if(_img != null)
+ {
+ measuredHeight = _img.getExplicitOrMeasuredHeight();
+ measuredWidth = _img.getExplicitOrMeasuredWidth();
+ }
+ }
+
+ /**
+ * Update the image on the display
+ */
+ override protected function updateDisplayList(unscaledWidth:Number, unscaledHeight:Number):void {
+ // Do this if and only iff the angle or the scale has changed. Otherwise we do not
+ // want to re-draw the image, only move it.
+ if(_img && (_realAngle != _angle || _realScale != _scale))
+ {
+ var contentWidth:Number = _img.getExplicitOrMeasuredWidth();
+ var contentHeight:Number= _img.getExplicitOrMeasuredHeight();
+
+ _img.setActualSize(contentWidth,contentHeight);
+
+ _realAngle = _angle;
+ _realScale = _scale;
+
+ // Distort the image
+ distortImage(_bitmap, _distortedShape, false);
+
+ // Create a reflection
+ createReflectionBitmap(_bitmap);
+ }
+ }
+
+ /**
+ * This function distorts the target according to its angle
+ * @param Bitmap bm - The bitmap of the object which is being distorted
+ * @param Shape target - The target where the distorted image is stored
+ * @param Boolean refelection - Is this target a reflection?
+ */
+ private function distortImage(bm:Bitmap, target:Shape, reflection:Boolean):void {
+ // Turn the angle into a value between 0 and 1
+ var k:Number = (Math.abs(_angle)/90);
+ k = Math.sqrt(k);
+
+ var vShear:Number = (_angle >= 0)? k*-perspectiveConstant : k*perspectiveConstant;
+ var verticalOffset:Number;
+
+ // How much is the width of the image scaled
+ var hScale:Number = 1 - k;
+
+ // How much lower/higher are the corners "further away" than the one "closer"
+ verticalOffset = (_img.getExplicitOrMeasuredHeight()*vShear);
+
+ // Is this image a reflection? If yes, then the vertical offset distortion is made in another direction
+ if(reflection)
+ verticalOffset *= -1;
+
+ // Initialize the distortion class
+ var distort:DistortImage = new DistortImage();
+ distort.container = target;
+ distort.target = bm;
+ distort.smooth = true;
+ distort.initialize( 5, 5, null );
+
+ // Distort the images to the left
+ if(_angle > 0) {
+ distort.setTransform(
+ 0,0,
+ _img.getExplicitOrMeasuredWidth()*hScale,-verticalOffset,
+ _img.getExplicitOrMeasuredWidth()*hScale,_img.getExplicitOrMeasuredHeight()+verticalOffset,
+ 0,_img.getExplicitOrMeasuredHeight()
+ );
+ }
+ // Distort the images to the right
+ else if(_angle < 0) {
+ distort.setTransform(
+ 0,verticalOffset,
+ _img.getExplicitOrMeasuredWidth()*hScale,0,
+ _img.getExplicitOrMeasuredWidth()*hScale,_img.getExplicitOrMeasuredHeight(),
+ 0,_img.getExplicitOrMeasuredHeight()-verticalOffset
+ );
+ }
+ distort.render();
+
+ // We use the original image as a reference, but do not want to show it on the screen
+ _img.visible = false;
+
+ // Align the distorted image correctly
+ var m:Matrix = target.transform.matrix;
+ m.tx = 0 - _img.width/2* hScale;
+ target.transform.matrix = m;
+ }
+
+ /**
+ * This function creates a reflection of the target object
+ * @param DisplayObject target
+ */
+ private function createReflectionBitmap(target:DisplayObject):void {
+ // Size of the fade
+ var fadeSize:Number = 0.4;
+
+ // Create a rectangle
+ var box:Rectangle = new Rectangle(0, 0, target.width, target.height);
+
+ // Create a matrix for the gradient
+ var gradientMatrix: Matrix = new Matrix();
+ // Create a shape object for the gradient
+ var gradientShape: Shape = new Shape();
+ // Apply a gradient on the matrix
+ gradientMatrix.createGradientBox(target.width, target.height * fadeSize, Math.PI/2, 0, target.height * (1.0 - fadeSize));
+
+ // Fill the shape with the gradient
+ gradientShape.graphics.beginGradientFill(GradientType.LINEAR, [0xFFFFFF, 0xFFFFFF], [0, 1], [0, 255], gradientMatrix);
+ gradientShape.graphics.drawRect(0, target.height * (1.0 - fadeSize), target.width, target.height * fadeSize);
+ gradientShape.graphics.endFill();
+
+ // Bitmap representation of the gradient
+ var gradientBm:BitmapData = new BitmapData(target.width, target.height, true, 0x00000000);
+ gradientBm.draw(gradientShape, new Matrix());
+
+ var targetBm:BitmapData = new BitmapData(target.width, target.height, true, 0x00000000);
+ targetBm.fillRect(box, 0x00000000);
+ targetBm.draw(target, new Matrix());
+
+ var reflectionData:BitmapData = new BitmapData(target.width, target.height, true, 0x00000000);
+ reflectionData.fillRect(box, 0x00000000);
+ reflectionData.copyPixels(targetBm, box, new Point(), gradientBm);
+
+ // Store the reflection
+ _reflectionBitmap = new Bitmap(reflectionData);
+
+ // Move the reflection to its correct position
+ var m:Matrix = _distortedShape.transform.matrix;
+ m.d = -1;
+ m.ty = _distortedShape.height*2;
+ _reflectionShape.transform.matrix = m;
+
+ // Distort the reflection
+ distortImage(_reflectionBitmap, _reflectionShape, true);
+
+ // Set a transparancy for the reflection
+ _reflectionShape.alpha = 0.4;
+
+ }
+
+ /**
+ * Get name
+ * @return String
+ */
+ public function get caption():String {
+ return this._caption;
+ }
+
+ /**
+ * Set name
+ * @param String name
+ */
+ public function set caption(caption:String):void {
+ this._caption = caption;
+ }
+
+ /**
+ * Get uri
+ * @return String
+ */
+ public function get uri():String {
+ return this._uri;
+ }
+
+ /**
+ * Set uri
+ * @param String uri
+ */
+ public function set uri(uri:String):void {
+ this._uri = uri;
+ }
+
+ /**
+ * Get the bitmap
+ * @return Bitmap
+ */
+ public function getBitmap():Bitmap {
+ this._bitmap = new Bitmap(this._bitmapData);
+ return this._bitmap;
+ }
+
+ /**
+ * Get the bitmap data
+ * @return BitmapData
+ */
+ public function getBitmapData():BitmapData {
+ return this._bitmapData;
+ }
+
+ /**
+ * Set the angle of the cover
+ * @param Number angle
+ */
+ public function set angle(angle:Number):void {
+ this._angle = angle;
+ invalidateDisplayList();
+ }
+
+ /**
+ * Get the angle of the cover
+ * @return Number
+ */
+ public function get angle():Number {
+ return this._angle;
+ }
+
+ /**
+ * Set the x position of the cover
+ * @param Number x
+ */
+ public function set xPos(x:Number):void {
+ this._x = x;
+ }
+
+ /**
+ * Get the x position of the cover
+ * @return Number
+ */
+ public function get xPos():Number {
+ return this._x;
+ }
+
+ /**
+ * Set the y position of the cover
+ * @param Number y
+ */
+ public function set yPos(y:Number):void {
+ this._y = y;
+ }
+
+ /**
+ * Get the y position of the cover
+ * @return Number
+ */
+ public function get yPos():Number {
+ return this._y;
+ }
+
+ /**
+ * Set the scale of the cover
+ * @param Number scale
+ */
+ public function set scale(scale:Number):void {
+ this._scale = scale;
+ }
+
+ /**
+ * Get the scale position of the cover
+ * @return Number
+ */
+ public function get scale():Number {
+ return this._scale;
+ }
+
+ }
+}
+ \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/Coverflow.as b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/Coverflow.as
new file mode 100644
index 0000000000..38d924b7dd
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/Coverflow.as
@@ -0,0 +1,598 @@
+package {
+ import flash.events.Event;
+ import flash.events.MouseEvent;
+ import flash.utils.Dictionary;
+ import flash.geom.Matrix;
+
+ import mx.controls.Alert;
+ import mx.controls.Image;
+ import mx.controls.HScrollBar;
+ import mx.core.UIComponent;
+ import mx.core.Application;
+ import mx.effects.AnimateProperty;
+ import mx.effects.easing.Quadratic;
+ import flash.external.ExternalInterface;
+ import mx.events.ScrollEvent;
+
+ public class Coverflow extends UIComponent {
+ // The scrollbar
+ private var _scrollbar:HScrollBar;
+
+ // _coverList is an array containing all the coverflow objects
+ private var _coverList:Array = new Array();
+
+ // Which element is selected
+ private var _selected:int = -1;
+
+ // Which element is currently being showed
+ private var _current:Number = 0;
+
+ // How much space (in pixels) is there between each cover
+ private var _coverSpacing:int = 40;
+
+ // A map between covers and their positions in the coverflow
+ private var _coverMap:Dictionary;
+
+ // The angle in which the child covers are distorted
+ private static const _angle:int = 35;
+
+ // The size of the child covers. The default value is 0.8 which means that
+ // the child covers are 80% of their maximum size
+ private static const _childScale:Number = 0.8;
+
+ // All covers are scaled to the maximum. These two variables tell us how
+ // big a cover can be in maximum
+ private var _maxWidth:int;
+ private var _maxHeight:int;
+
+ // An object which takes care of animations
+ private var _animation:AnimateProperty;
+
+ // An object for event handling
+ private var _eventHandler:EventHandler = new EventHandler();
+
+ // The unique identifier of this instance of the widget
+ private var _pid:String;
+
+ // Has the cover list's content changed?
+ private var _listChanged:Boolean = false;
+
+ /**
+ * Constructor
+ */
+ public function Coverflow():void {
+ super();
+
+ // Initialize the scrollbar
+ _scrollbar = new HScrollBar();
+ _scrollbar.x = 0;
+ // Add an action listener to the scrollbar which detects when the
+ // scrollbar's position has been changed. This event should also
+ // change the selected cover's value
+ _scrollbar.addEventListener(ScrollEvent.SCROLL, function ():void { selectedCover = Math.round(_scrollbar.scrollPosition); });
+
+ // Maximize the size of the component
+ this.percentHeight = 100;
+ this.percentWidth = 100;
+
+ }
+
+ /**
+ * This function is called when the flash has finished loading. This
+ * function will intialize the communication interface between flash
+ * and GWT.
+ */
+ private function init():void {
+ // Are we even able to initialize a external communication interface?
+ if (ExternalInterface.available) {
+ // These two methods are made available for javascript (they
+ // can be directly called within javascript code)
+ ExternalInterface.addCallback("addCover", this.addCover);
+ ExternalInterface.addCallback("selectCover", this.externalSetCover);
+ ExternalInterface.addCallback("setBackgroundColor", this.setBackgroundColor);
+ ExternalInterface.addCallback("toggleScrollbarVisibility", this.toggleScrollbarVisibility);
+ ExternalInterface.addCallback("removeCover", this.removeCover);
+
+ // Try to call a javascript function
+ try {
+ // The function we want to call is getCovers. It tells javascript
+ // that the flash is now ready to accept information of the covers.
+ // The name of the function we're about to call is dynamic, meaning
+ // it is unique for every instance of this widget
+ ExternalInterface.call("itmill.coverflow['getCovers_" + _pid + "']");
+ } catch (error:SecurityError) {
+ Alert.show("A SecurityError occurred: " + error.message + "\n");
+ } catch (error:Error) {
+ Alert.show("An Error occurred: " + error.message + "\n");
+ }
+ } else {
+ Alert.show("External interface is not available for this container.");
+ }
+ }
+
+ /**
+ * Adds a new cover to the coverflow list
+ * @param String name
+ * @param String uri
+ */
+ private function addCover(name:String, uri:String):void {
+ // Create an instance of the cover object
+ var cover:Cover = new Cover(name.toString(), uri.toString());
+
+ // Mark the list as being modified
+ _listChanged = true;
+
+ // Load the image
+ cover.loadImage();
+ // Add the cover to our array
+ _coverList.push(cover);
+ commitProperties();
+ }
+
+ /**
+ * Measure the size of this component
+ */
+ override protected function measure():void {
+ super.measure();
+
+ // What is the maximum size of a cover? It is either 3/4 of the height of the component
+ // or 1/3 of the width - which ever is smaller.
+ _maxHeight = _maxWidth = Math.round(Math.min(this.width/3,this.height/4*3));
+ }
+
+ /**
+ * Creates the child elements (covers)
+ */
+ override protected function createChildren():void {
+ super.createChildren();
+
+ for(var i:int = 0; i < _coverList.lenght; i++)
+ addChild(_coverList[i]);
+
+ invalidateDisplayList();
+ }
+
+ /**
+ * Something has changed
+ */
+ override protected function commitProperties():void {
+ // Remove all old covers
+ for(i = numChildren-1;i>=0;i--) {
+ removeChildAt(numChildren-1);
+ }
+
+ // Create a new mapping between the covers and their position
+ _coverMap = new Dictionary(true);
+
+ // Loop through the coverlist array
+ for(var i:int = 0;i<_coverList.length;i++) {
+ var cover:Cover = _coverList[i];
+ // Make sure the cover is loaded
+ cover.loadImage();
+
+ // Add an event listener to the cover. We want to know when
+ // a cover is being clicked.
+ cover.addEventListener(MouseEvent.CLICK,selectEvent,false,0,true);
+ // Add the cover to the flash movie
+ addChildAt(cover,i);
+
+ // Add this cover to the map
+ _coverMap[cover] = i;
+ }
+
+ // Add the scrollbar on top of all other elements
+ addChild(_scrollbar);
+
+ // Update the scrollbar's details
+ invalidateScrollbar();
+
+ // update screen
+ invalidateDisplayList();
+ }
+
+ /**
+ * This function draws the actual content of the flash movie
+ */
+ override protected function updateDisplayList(unscaledWidth:Number, unscaledHeight:Number):void {
+ if(_selected < 0) {
+ selectedCover = 0;
+ return;
+ }
+
+ // Measure the size of the component
+ measure();
+ // Update the scrollbar's details
+ invalidateScrollbar();
+
+ // Check if there are any new covers added to our array
+ if(_listChanged) {
+ _eventHandler.dispatchEvent(new Event(EventHandler.DATA_CHANGED));
+ _listChanged = false;
+ }
+
+ // If no covers exist, then do nothing
+ if(_coverList.length == 0)
+ return;
+
+
+ var cover:Cover;
+ var index:int = 0;
+ var m:Matrix;
+
+ // Loop through all covers
+ for(var i:int=0; i < _coverList.length; i++) {
+ cover = _coverList[i];
+
+ // Calculate in which layer position the cover should be
+ index = (i <= Math.floor(_current))? i : _coverList.length-1-i;
+
+ // Set the real size of the cover
+ cover.setActualSize(cover.getExplicitOrMeasuredWidth(),cover.getExplicitOrMeasuredHeight());
+
+ // Calculate in which angle the cover should be at this specific moment
+ calculateAngle(cover, i);
+
+ // Set the covers layer position
+ setChildIndex(cover,index);
+
+ // Calculate the cover's size (scale)
+ calculateScale(cover, i);
+
+ // ...and finally calculate its position (horizontal position, that is)
+ calculatePosition(cover, i);
+
+ // Resize the cover according to its scale value
+ m = cover.transform.matrix;
+ m.a = m.d = cover.scale;
+ cover.transform.matrix = m;
+
+ // Move the cover to its correct position
+ cover.move(cover.xPos, cover.yPos);
+
+ }
+ // Set selected cover on top of all other covers
+ setChildIndex(_coverList[Math.floor(_current)],_coverList.length-1);
+ }
+
+ /**
+ * This function calulcates the position of a cover at any given moment.
+ * @param Cover c - Which cover is being processed
+ * @param int index - what is the cover's index (horizontal order number)
+ */
+ private function calculatePosition(c:Cover, index:int):void {
+ // All covers are aligned so, that their bottom is at 3/4's height of the
+ // actual flash movie
+ c.yPos = Math.round(unscaledHeight/4*3-c.getExplicitOrMeasuredHeight()*c.scale);
+
+ // Calculate the position difference between the currently selected item and
+ // this items index
+ var diff:Number = _current-index;
+
+ // We want to know the previous and next positions of the item
+ var prevPos:Number;
+ var nextPos:Number;
+
+ // We are currently processing a cover which comes from the left to the center (selected)
+ if(index == Math.floor(_current)) {
+ // Calculate the previous position
+ prevPos = unscaledWidth/2 - _maxWidth/2 - _coverSpacing;
+ // The next position is in the center of the screen
+ nextPos = unscaledWidth/2;
+ // Now calculate in which state of the animation we are
+ c.xPos = nextPos-diff*(nextPos-prevPos);
+ }
+ // Same as above, except now we come from right to center
+ else if(index == Math.ceil(_current)) {
+ prevPos = unscaledWidth/2;
+ nextPos = unscaledWidth/2 +_maxWidth/2 + _coverSpacing;
+ c.xPos = prevPos-diff*(nextPos-prevPos);
+ }
+ // Child covers to the left
+ else if(index < Math.floor(_current)){
+ c.xPos = unscaledWidth/2 - _maxWidth/2 - diff*_coverSpacing;
+ }
+ // ..and child covers to the right
+ else {
+ c.xPos = unscaledWidth/2 +_maxWidth/2 - diff*_coverSpacing;
+ }
+ }
+
+ /**
+ * Calulcates the angle of a cover at any given moment
+ * @param Cover c
+ * @param int index
+ */
+ private function calculateAngle(c:Cover, index:int):void {
+ // This function has the same principle as the the function above
+
+ var diff:Number;
+
+ if(index == Math.floor(_current)) {
+ diff = _current-Math.floor(_current);
+ c.angle = _angle*diff;
+ }
+ else if(index == Math.ceil(_current)) {
+ diff = _current-Math.ceil(_current);
+ c.angle = _angle*diff;
+ }
+ else if(index < Math.floor(_current))
+ c.angle = _angle;
+ else
+ c.angle = -_angle;
+ }
+
+ /**
+ * Calculates the size (scale) of a cover at any given moment
+ * @param Cover c
+ * @param int index
+ */
+ private function calculateScale(c:Cover, index:int):void {
+ // Almost same as calculatePosition, except that now we are
+ // calculating the scaled size of the cover
+
+ var diff:Number = _current-index;
+ var scalePrev:Number;
+ var scaleNext:Number;
+
+ if(index == Math.floor(_current)) {
+ scalePrev = maxScale(c.width, c.height)*_childScale;
+ scaleNext = maxScale(c.width, c.height);
+ c.scale = scaleNext-diff*(scaleNext-scalePrev);
+ }
+ else if(index == Math.ceil(_current)) {
+ scalePrev = maxScale(c.width, c.height);
+ scaleNext = maxScale(c.width, c.height)*_childScale;
+ c.scale = scalePrev-diff*(scaleNext-scalePrev);
+ }
+ else
+ c.scale = maxScale(c.width, c.height)*_childScale;
+
+
+ }
+
+ /**
+ * Calculate the maximum scale of a cover
+ * @param Number w - The actual width of the cover
+ * @param Number h - The actual height of the cover
+ * @return Number
+ */
+ private function maxScale(w:Number, h:Number):Number {
+ // The width of the cover is bigger than the height. This means that
+ // the width will be the restricting factor of this cover's size.
+ // Therefore we calculate the maximum size of this cover (result given
+ // as a scale)
+ if(w > h)
+ return _maxWidth/w;
+
+ // Height is bigger than width. Same logic as above.
+ else
+ return _maxHeight/h;
+ }
+
+ /**
+ * What happens when a user clicks on a cover?
+ * @param MouseEvent e
+ */
+ private function selectEvent(e:MouseEvent):void {
+ // Use the mapping between the covers and their positions to
+ // find out which is the index of the cover that was clicked.
+ // Set this index as the new selected cover.
+ selectedCover = _coverMap[e.currentTarget];
+ }
+
+ /**
+ * Set the selected cover
+ * @param int index
+ */
+ public function set selectedCover(index:int):void {
+ // The selected cover is already selected. Do nothing
+ if(index == _selected)
+ return;
+
+ // Validate the index, make sure it's within the coverList's range
+ if(index >= 0 && index < _coverList.length) {
+ _selected = index;
+
+ // Animate the changes
+ animateChange();
+ }
+ }
+
+ /**
+ * Returns the selected cover
+ * @return int
+ */
+ public function get selectedCover():int {
+ return _selected;
+ }
+
+ /**
+ * Set the current cover (which cover is actually being shown at this very moment.
+ * Note that the value can be a decimal value, because the selected cover can be for
+ * example 3.2, which means that it has moved 20% between the positions 3 and 4.
+ * @param Number i
+ */
+ public function set current(i:Number):void {
+ // If ExternalInterface is available and the current cover
+ // is same as the currently selected cover (in other words,
+ // the animation has finished), then send the selected cover's
+ // key to GWT which will then forward it to the server.
+ if (ExternalInterface.available && i == _selected) {
+ ExternalInterface.call("itmill.coverflow['setCurrent_" + _pid + "']",_coverList[_selected].caption);
+
+ // Send an event which notifies the scrollbar that the selected cover has changed.
+ _eventHandler.dispatchEvent(new Event(EventHandler.CURRENT_CHANGED));
+ }
+
+ // Update value
+ _current = i;
+ // Update display
+ invalidateDisplayList();
+ }
+
+ /**
+ * Get the value of current
+ * @return Number
+ */
+ public function get current():Number {
+ return _current;
+ }
+
+ /**
+ * This is a function where an external source can set the selected cover
+ * (in our case, it could be another widget which is connected to our
+ * coverflow).
+ *
+ * @param String index
+ */
+ private function externalSetCover(id:String):void {
+ var cover:Cover;
+ for(var i:int=0; i < _coverList.length; i++) {
+ cover = _coverList[i];
+
+ if(cover.caption == id) {
+ selectedCover = i;
+ break;
+ }
+ }
+ }
+
+ /**
+ * With this function we can remove any cover from the cover flow in run time
+ *
+ * @param String id
+ */
+ private function removeCover(id:String):void {
+ var cover:Cover;
+
+ // Loop through all covers and search the correct one
+ for(var i:int=0; i < _coverList.length; i++) {
+ cover = _coverList[i];
+
+ // If the cover's name matches with the given id, then delete it
+ if(cover.caption == id) {
+ // First we will however check if we are removing the last
+ // which is also currently selected
+ if(i == _coverList.length-1 && i == _selected) {
+ // Select the previous cover
+ selectedCover = i-1;
+
+ // Jump to the end of the animation
+ if(_animation != null && _animation.isPlaying)
+ _animation.end();
+ }
+
+ // Mark the list as having changes
+ _listChanged = true;
+
+ // Remove the cover from the list
+ _coverList.splice(i,1);
+
+ // Update
+ commitProperties();
+ break;
+ }
+ }
+ }
+
+ /**
+ * Gives a references to the event handler
+ * @return EventHandler
+ */
+ public function get eventHandler():EventHandler {
+ return _eventHandler;
+ }
+
+ /**
+ * Tells us how many covers there are currently added
+ * @return int
+ */
+ public function get coverCount():int {
+ return _coverList.length;
+ }
+
+ /**
+ * Sets the unique identifier of this widget. This is used to create
+ * the dynamic function names in the external interface calls.
+ * @param String p
+ */
+ public function set pid(p:String):void {
+ if(_pid == null) {
+ _pid = p;
+ init();
+ }
+ }
+
+ /**
+ * Set the background color of the coverflow
+ * @param String gradientStart
+ * @param String gradientEnd
+ */
+ public function setBackgroundColor(gradientStart:String, gradientEnd:String):void {
+ Application.application.setStyle('backgroundGradientColors', [gradientStart, gradientEnd]);
+ }
+
+ /**
+ * Make sure the scrollbar is up-to-date, both in size and position wise
+ */
+ private function invalidateScrollbar():void {
+ _scrollbar.width = unscaledWidth;
+ _scrollbar.y = unscaledHeight-_scrollbar.getExplicitOrMeasuredHeight()/2;
+ _scrollbar.maxScrollPosition = coverCount-1;
+ _scrollbar.scrollPosition = _selected;
+ _scrollbar.pageSize = 1;
+ _scrollbar.invalidateDisplayList();
+ }
+
+ /**
+ * Change the visibility status of the scrollbar
+ * @param String visibility
+ */
+ public function toggleScrollbarVisibility(visibility:String):void {
+ // Input is a string, because javascript can call directly on this function
+ if(visibility == "false")
+ _scrollbar.visible = false;
+ else
+ _scrollbar.visible = true;
+ }
+
+ /**
+ * Animate the state changes
+ */
+ private function animateChange():void {
+ // If there already is an animation in process, stop it (jump to the end) and start the new one.
+ if(_animation != null && _animation.isPlaying)
+ _animation.end();
+
+ // Set up a new animation. We want the "current" value go to the "_selected" value
+ _animation = new AnimateProperty(this);
+ _animation.property = "current";
+ _animation.toValue = _selected;
+ _animation.target = this;
+
+ // What is the duration of the animation? We don't want the animation to go too fast or too slow.
+ // An animation where the selected cover is changed with only one position should be relatively
+ // slow, so that we actually can se an animation. Therefore we've set a minimum length of the
+ // animation to 400 ms. If we jump over several covers, then we want the animation to be a
+ // bit longer, so that the animation would be smoother. Therefore we reserve 200ms between every
+ // cover change. Select which ever is bigger, 400ms or difference between current and selected
+ // cover * 200ms.
+ var duration:int = Math.max(400,Math.abs(Math.ceil(_current)-_selected)*200);
+
+ // If we do long jumps (for example by using the slider), we don't want the animation to go
+ // too long. Therefore we're setting a maximum length for the animation, in this case 2 seconds.
+ duration = Math.min(2000,duration);
+ _animation.duration = duration;
+
+ // We don't want the animation to be linear, we want it to slow down in the end
+ _animation.easingFunction = mx.effects.easing.Quadratic.easeOut;
+
+ // Start the animation
+ _animation.play();
+ }
+
+ }
+
+
+}
+ \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/EventHandler.as b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/EventHandler.as
new file mode 100644
index 0000000000..f9d25be24d
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/EventHandler.as
@@ -0,0 +1,16 @@
+package {
+ import flash.events.EventDispatcher;
+ import flash.events.Event;
+
+ /**
+ * This is a simple class created for event handling. Basically it just
+ * dispatches some custom events related to state changes within the coverflow.
+ */
+ public class EventHandler extends EventDispatcher {
+ // Events
+ public static var DATA_CHANGED:String = "data_changed";
+ public static var CURRENT_CHANGED:String = "current_changed";
+
+
+ }
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/coverflowflash.mxml b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/coverflowflash.mxml
new file mode 100644
index 0000000000..612e8ba52f
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/coverflowflash.mxml
@@ -0,0 +1,34 @@
+<?xml version="1.0" encoding="utf-8"?>
+
+<mx:Application
+ xmlns:mx="http://www.adobe.com/2006/mxml"
+ horizontalAlign="center" verticalAlign="middle"
+ creationComplete="{ init(); }"
+ xmlns:custom="*"
+ paddingLeft="0"
+ paddingTop="10"
+ paddingBottom="0"
+ paddingRight="0"
+ >
+
+ <mx:Script>
+ <![CDATA[
+ import mx.controls.Alert;
+
+ private var coverflow:Coverflow = new Coverflow();
+
+ private function init():void {
+ coverflow.setBackgroundColor(Application.application.parameters.bgS, Application.application.parameters.bgE);
+ coverflow.toggleScrollbarVisibility(Application.application.parameters.sbVis);
+ coverflow.pid = Application.application.parameters.pid;
+
+ container.addChild(coverflow);
+
+ }
+
+ ]]>
+ </mx:Script>
+
+ <mx:Box width="100%" height="100%" id="container" />
+</mx:Application>
+
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/DistortImage.as b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/DistortImage.as
new file mode 100644
index 0000000000..101e9e217c
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/DistortImage.as
@@ -0,0 +1,306 @@
+/*
+# ***** BEGIN LICENSE BLOCK *****
+Copyright the original author or authors.
+Licensed under the MOZILLA PUBLIC LICENSE, Version 1.1 (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.mozilla.org/MPL/MPL-1.1.html
+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.
+
+# ***** END LICENSE BLOCK *****
+*/
+
+
+/*
+****************************
+* From a first idea and first implementation of Andre Michelle www.andre-michelle.com
+* @version 2.0
+* @author Thomas Pfeiffer - kiroukou - http://www.thomas-pfeiffer.info
+* @author Richard Lester - RichL
+* @author Didier Brun - foxy - http://www.foxaweb.com
+* @author Alex Uhlmann
+* @website: http://sandy.media-box.net
+* @description: Tesselate a movieclip into several triangles to allow free transform distorsion.
+*/
+package sandy.util
+{
+ import flash.display.BitmapData;
+ import flash.display.DisplayObject;
+ import flash.display.Shape;
+ import flash.display.Sprite;
+ import flash.geom.Matrix;
+ import flash.geom.Rectangle;
+
+ public class DistortImage
+ {
+ public var smooth : Boolean;
+ // -- texture to distort
+ public var texture:BitmapData;
+ // -- container Sprite or Shape : the display object containing the distorted picture drawn via graphics.
+
+ private var _container:Object;
+ public function get container() : Object
+ {
+ return _container;
+ }
+
+ public function set container( value : Object ) : void
+ {
+ if( ( value is Shape ) || ( value is Sprite ) )
+ {
+ _container = value;
+ }
+ else
+ {
+ throw new Error('container must be flash.display.Shape or flash.display.Sprite');
+ }
+ }
+
+ // -- target Object : either a BitmapData or a sprite
+ public var target:Object;
+ // -- arrays of differents datas types
+ public var points:Array;
+
+ /////////////////////////
+ /// PRIVATE PROPERTIES //
+ /////////////////////////
+ private var offsetRect:Rectangle;
+ private var _w:Number;
+ private var _h:Number;
+ private var _xMin:Number;
+ private var _xMax:Number;
+ private var _yMin:Number
+ private var _yMax:Number;
+ // -- picture segmentation properties
+ private var _hseg:Number;
+ private var _vseg:Number;
+ private var _hsLen:Number;
+ private var _vsLen:Number;
+ private var _tri:Array;
+ private var _aMcs:Array;
+
+ public function DistortImage()
+ {
+ smooth = true;
+ }
+
+ /*
+ * @param vseg Number : the vertical precision
+ * @param hseg Number : the horizontal precision
+ * @param offsetRect Rectangle : optional, the real bounds to use.
+ * @throws: An error if target property isn't a BitmapData or a DisplayObject
+ */
+ public function initialize( vseg: Number, hseg: Number, offsetRect: Rectangle = null ) : void
+ {
+ if( target is BitmapData )
+ {
+ texture = target as BitmapData;
+ _w = texture.width;
+ _h = texture.height;
+ }
+ else if( target is DisplayObject )
+ {
+ renderVector( offsetRect );
+ }
+ else
+ {
+ throw new Error('target must be flash.display.BitmapData or flash.display.DisplayObject');
+ }
+ _vseg = vseg || 0;
+ _hseg = hseg || 0;
+
+ // --
+ _aMcs = new Array();
+ points = new Array();
+ _tri = new Array();
+ // --
+ __init();
+ }
+
+ public function render() : void
+ {
+ __render();
+ }
+
+ /**
+ * setTransform
+ *
+ * @param x0 Number the horizontal coordinate of the first point
+ * @param y0 Number the vertical coordinate of the first point
+ * @param x1 Number the horizontal coordinate of the second point
+ * @param y1 Number the vertical coordinate of the second point
+ * @param x2 Number the horizontal coordinate of the third point
+ * @param y2 Number the vertical coordinate of the third point
+ * @param x3 Number the horizontal coordinate of the fourth point
+ * @param y3 Number the vertical coordinate of the fourth point
+ *
+ * @description : Distort the bitmap to adjust it to those points.
+ */
+ public function setTransform( x0:Number , y0:Number ,
+ x1:Number , y1:Number ,
+ x2:Number , y2:Number ,
+ x3:Number , y3:Number ): void
+ {
+ var w:Number = _w;
+ var h:Number = _h;
+ var dx30:Number = x3 - x0;
+ var dy30:Number = y3 - y0;
+ var dx21:Number = x2 - x1;
+ var dy21:Number = y2 - y1;
+ var l:int = points.length;
+ while( --l > -1 )
+ {
+ var point:SandyPoint = points[ l ];
+ var gx:Number = ( point.x - _xMin ) / w;
+ var gy:Number = ( point.y - _yMin ) / h;
+ var bx:Number = x0 + gy * ( dx30 );
+ var by:Number = y0 + gy * ( dy30 );
+
+ point.sx = bx + gx * ( ( x1 + gy * ( dx21 ) ) - bx );
+ point.sy = by + gx * ( ( y1 + gy * ( dy21 ) ) - by );
+ }
+ __render();
+ }
+
+
+
+ /////////////////////////
+ /// PRIVATE METHODS ///
+ /////////////////////////
+
+ private function renderVector( offsetRect: Rectangle = null ) : void
+ {
+ var vector : DisplayObject = target as DisplayObject;
+ if( offsetRect != null )
+ {
+ texture = new BitmapData( offsetRect.width , offsetRect.height, true, 0x00000000 );
+ }
+ else
+ {
+ texture = new BitmapData( vector.width , vector.height, true, 0x00000000 );
+ offsetRect = new Rectangle( 0, 0, texture.width, texture.height );
+ }
+
+ var m : Matrix = new Matrix();
+ m.translate( offsetRect.x * -1, offsetRect.y * -1 );
+ texture.draw( vector, m );
+ container.transform.matrix.translate( vector.transform.matrix.tx, vector.transform.matrix.ty );
+ _w = offsetRect.width;
+ _h = offsetRect.height;
+ }
+
+ private function __init(): void
+ {
+ points = new Array();
+ _tri = new Array();
+ var ix:int, iy:int;
+ var w2: Number = _w / 2;
+ var h2: Number = _h / 2;
+ _xMin = _yMin = 0;
+ _xMax = _w; _yMax = _h;
+ _hsLen = _w / ( _hseg + 1 );
+ _vsLen = _h / ( _vseg + 1 );
+ var x:Number, y:Number;
+ var p0:SandyPoint, p1:SandyPoint, p2:SandyPoint;
+
+ // -- we create the points
+ for ( ix = 0 ; ix < _hseg + 2 ; ix++ )
+ {
+ for ( iy = 0 ; iy < _vseg + 2 ; iy++ )
+ {
+ x = ix * _hsLen;
+ y = iy * _vsLen;
+ points.push( new SandyPoint( x, y, x, y ) );
+ }
+ }
+ // -- we create the triangles
+ for ( ix = 0 ; ix < _vseg + 1 ; ix++ )
+ {
+ for ( iy = 0 ; iy < _hseg + 1 ; iy++ )
+ {
+ p0 = points[ iy + ix * ( _hseg + 2 ) ];
+ p1 = points[ iy + ix * ( _hseg + 2 ) + 1 ];
+ p2 = points[ iy + ( ix + 1 ) * ( _hseg + 2 ) ];
+ __addTriangle( p0, p1, p2 );
+ // --
+ p0 = points[ iy + ( ix + 1 ) * ( _vseg + 2 ) + 1 ];
+ p1 = points[ iy + ( ix + 1 ) * ( _vseg + 2 ) ];
+ p2 = points[ iy + ix * ( _vseg + 2 ) + 1 ];
+ __addTriangle( p0, p1, p2 );
+ }
+ }
+ }
+
+ private function __addTriangle( p0:SandyPoint, p1:SandyPoint, p2:SandyPoint ):void
+ {
+ var u0:Number, v0:Number, u1:Number, v1:Number, u2:Number, v2:Number;
+ var tMat:Matrix = new Matrix();
+ // --
+ u0 = p0.x; v0 = p0.y;
+ u1 = p1.x; v1 = p1.y;
+ u2 = p2.x; v2 = p2.y;
+ tMat.tx = -v0*(_w / (v1 - v0));
+ tMat.ty = -u0*(_h / (u2 - u0));
+ tMat.a = tMat.d = 0;
+ tMat.b = _h / (u2 - u0);
+ tMat.c = _w / (v1 - v0);
+ // --
+ _tri.push( new Triangle( p0, p1, p2, tMat ) );
+ }
+
+ private function __render(): void
+ {
+ var vertices: Array;
+ var p0:SandyPoint, p1:SandyPoint, p2:SandyPoint;
+ var x0:Number, y0:Number;
+ var ih:Number = 1/_h, iw:Number = 1/_w;
+ var c:Object = container; c.graphics.clear();
+ var a:Triangle;
+ var sM:Matrix = new Matrix();
+ var tM:Matrix = new Matrix();
+ //--
+ var l:int = _tri.length;
+ while( --l > -1 )
+ {
+ a = _tri[ l ];
+ p0 = a.p0;
+ p1 = a.p1;
+ p2 = a.p2;
+ tM = a.tMat;
+ // --
+ sM.a = ( p1.sx - ( x0 = p0.sx ) ) * iw;
+ sM.b = ( p1.sy - ( y0 = p0.sy ) ) * iw;
+ sM.c = ( p2.sx - x0 ) * ih;
+ sM.d = ( p2.sy - y0 ) * ih;
+ sM.tx = x0;
+ sM.ty = y0;
+ // --
+ sM = __concat( sM, tM );
+
+ c.graphics.beginBitmapFill( texture, sM, false, smooth );
+ c.graphics.moveTo( x0, y0 );
+ c.graphics.lineTo( p1.sx, p1.sy );
+ c.graphics.lineTo( p2.sx, p2.sy );
+ c.graphics.endFill();
+ }
+ }
+
+ private function __concat( m1:Matrix, m2:Matrix ):Matrix
+ {
+ //Relies on the original triangles being right angled with p0 being the right angle.
+ //Therefore a = d = zero (before and after invert)
+ var mat : Matrix = new Matrix();
+ mat.a = m1.c * m2.b;
+ mat.b = m1.d * m2.b;
+ mat.c = m1.a * m2.c;
+ mat.d = m1.b * m2.c;
+ mat.tx = m1.a * m2.tx + m1.c * m2.ty + m1.tx;
+ mat.ty = m1.b * m2.tx + m1.d * m2.ty + m1.ty;
+ return mat;
+ }
+ }
+}
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/SandyPoint.as b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/SandyPoint.as
new file mode 100644
index 0000000000..7d5c52ca0c
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/SandyPoint.as
@@ -0,0 +1,17 @@
+package sandy.util
+{
+ import flash.geom.Point;
+
+ public class SandyPoint extends Point
+ {
+ public var sx : Number;
+ public var sy : Number;
+
+ public function SandyPoint( x : Number, y : Number, sx : Number, sy : Number )
+ {
+ super( x, y );
+ this.sx = sx;
+ this.sy = sy;
+ }
+ }
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/Triangle.as b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/Triangle.as
new file mode 100644
index 0000000000..8f08fb6c76
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/flex/sandy/util/Triangle.as
@@ -0,0 +1,20 @@
+package sandy.util
+{
+ import flash.geom.Matrix;
+
+ public class Triangle
+ {
+ public var p0 : SandyPoint;
+ public var p1 : SandyPoint;
+ public var p2 : SandyPoint;
+ public var tMat : Matrix;
+
+ public function Triangle( p0 : SandyPoint, p1 : SandyPoint, p2 : SandyPoint, tMat : Matrix )
+ {
+ this.p0 = p0;
+ this.p1 = p1
+ this.p2 = p2;
+ this.tMat = tMat;
+ }
+ }
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/coverflow/gwt/public/coverflowflash.swf b/src/com/itmill/toolkit/demo/coverflow/gwt/public/coverflowflash.swf
new file mode 100644
index 0000000000..f3c75f4fe3
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/coverflow/gwt/public/coverflowflash.swf
Binary files differ
diff --git a/src/com/itmill/toolkit/demo/m-bullet-blue.gif b/src/com/itmill/toolkit/demo/m-bullet-blue.gif
deleted file mode 100644
index fa6b38b4c9..0000000000
--- a/src/com/itmill/toolkit/demo/m-bullet-blue.gif
+++ /dev/null
Binary files differ
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/AddressBookApplication.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/AddressBookApplication.java
new file mode 100644
index 0000000000..b2c38170d6
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/AddressBookApplication.java
@@ -0,0 +1,244 @@
+/*
+@ITMillApache2LicenseForJavaFiles@
+ */
+
+package com.itmill.toolkit.demo.tutorial.addressbook;
+
+import com.itmill.toolkit.Application;
+import com.itmill.toolkit.data.Item;
+import com.itmill.toolkit.data.Property;
+import com.itmill.toolkit.data.Property.ValueChangeEvent;
+import com.itmill.toolkit.data.Property.ValueChangeListener;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.PersonContainer;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.SearchFilter;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.HelpWindow;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.ListView;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.NavigationTree;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.PersonForm;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.PersonList;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.SearchView;
+import com.itmill.toolkit.demo.tutorial.addressbook.ui.SharingOptions;
+import com.itmill.toolkit.event.ItemClickEvent;
+import com.itmill.toolkit.event.ItemClickEvent.ItemClickListener;
+import com.itmill.toolkit.terminal.ThemeResource;
+import com.itmill.toolkit.ui.Alignment;
+import com.itmill.toolkit.ui.Button;
+import com.itmill.toolkit.ui.Component;
+import com.itmill.toolkit.ui.Embedded;
+import com.itmill.toolkit.ui.HorizontalLayout;
+import com.itmill.toolkit.ui.SplitPanel;
+import com.itmill.toolkit.ui.VerticalLayout;
+import com.itmill.toolkit.ui.Window;
+import com.itmill.toolkit.ui.Button.ClickEvent;
+import com.itmill.toolkit.ui.Button.ClickListener;
+import com.itmill.toolkit.ui.Window.Notification;
+
+public class AddressBookApplication extends Application implements
+ ClickListener, ValueChangeListener, ItemClickListener {
+
+ private NavigationTree tree = new NavigationTree(this);
+
+ private Button newContact = new Button("Add contact");
+ private Button search = new Button("Search");
+ private Button share = new Button("Share");
+ private Button help = new Button("Help");
+ private SplitPanel horizontalSplit = new SplitPanel(
+ SplitPanel.ORIENTATION_HORIZONTAL);
+
+ // Lazyly created ui references
+ private ListView listView = null;
+ private SearchView searchView = null;
+ private PersonList personList = null;
+ private PersonForm personForm = null;
+ private HelpWindow helpWindow = null;
+ private SharingOptions sharingOptions = null;
+
+ private PersonContainer dataSource = PersonContainer.createWithTestData();
+
+ @Override
+ public void init() {
+ buildMainLayout();
+ setMainComponent(getListView());
+ }
+
+ private void buildMainLayout() {
+ setMainWindow(new Window("Address Book Demo application"));
+
+ setTheme("contacts");
+
+ VerticalLayout layout = new VerticalLayout();
+ layout.setSizeFull();
+
+ layout.addComponent(createToolbar());
+ layout.addComponent(horizontalSplit);
+ layout.setExpandRatio(horizontalSplit, 1);
+
+ horizontalSplit.setSplitPosition(200, SplitPanel.UNITS_PIXELS);
+ horizontalSplit.setFirstComponent(tree);
+
+ getMainWindow().setLayout(layout);
+ }
+
+ private HorizontalLayout createToolbar() {
+ HorizontalLayout lo = new HorizontalLayout();
+ lo.addComponent(newContact);
+ lo.addComponent(search);
+ lo.addComponent(share);
+ lo.addComponent(help);
+
+ search.addListener((ClickListener) this);
+ share.addListener((ClickListener) this);
+ help.addListener((ClickListener) this);
+ newContact.addListener((ClickListener) this);
+
+ search.setIcon(new ThemeResource("icons/32/folder-add.png"));
+ share.setIcon(new ThemeResource("icons/32/users.png"));
+ help.setIcon(new ThemeResource("icons/32/help.png"));
+ newContact.setIcon(new ThemeResource("icons/32/document-add.png"));
+
+ lo.setMargin(true);
+ lo.setSpacing(true);
+
+ lo.setStyleName("toolbar");
+
+ lo.setWidth("100%");
+
+ Embedded em = new Embedded("", new ThemeResource("images/logo.png"));
+ lo.addComponent(em);
+ lo.setComponentAlignment(em, Alignment.MIDDLE_RIGHT);
+ lo.setExpandRatio(em, 1);
+
+ return lo;
+ }
+
+ private void setMainComponent(Component c) {
+ horizontalSplit.setSecondComponent(c);
+ }
+
+ /*
+ * View getters exist so we can lazily generate the views, resulting in
+ * faster application startup time.
+ */
+ private ListView getListView() {
+ if (listView == null) {
+ personList = new PersonList(this);
+ personForm = new PersonForm(this);
+ listView = new ListView(personList, personForm);
+ }
+ return listView;
+ }
+
+ private SearchView getSearchView() {
+ if (searchView == null) {
+ searchView = new SearchView(this);
+ }
+ return searchView;
+ }
+
+ private HelpWindow getHelpWindow() {
+ if (helpWindow == null) {
+ helpWindow = new HelpWindow();
+ }
+ return helpWindow;
+ }
+
+ private SharingOptions getSharingOptions() {
+ if (sharingOptions == null) {
+ sharingOptions = new SharingOptions();
+ }
+ return sharingOptions;
+ }
+
+ public PersonContainer getDataSource() {
+ return dataSource;
+ }
+
+ public void buttonClick(ClickEvent event) {
+ final Button source = event.getButton();
+
+ if (source == search) {
+ showSearchView();
+ } else if (source == help) {
+ showHelpWindow();
+ } else if (source == share) {
+ showShareWindow();
+ } else if (source == newContact) {
+ addNewContanct();
+ }
+ }
+
+ private void showHelpWindow() {
+ getMainWindow().addWindow(getHelpWindow());
+ }
+
+ private void showShareWindow() {
+ getMainWindow().addWindow(getSharingOptions());
+ }
+
+ private void showListView() {
+ setMainComponent(getListView());
+ }
+
+ private void showSearchView() {
+ setMainComponent(getSearchView());
+ }
+
+ public void valueChange(ValueChangeEvent event) {
+ Property property = event.getProperty();
+ if (property == personList) {
+ Item item = personList.getItem(personList.getValue());
+ if (item != personForm.getItemDataSource()) {
+ personForm.setItemDataSource(item);
+ }
+ }
+ }
+
+ public void itemClick(ItemClickEvent event) {
+ if (event.getSource() == tree) {
+ Object itemId = event.getItemId();
+ if (itemId != null) {
+ if (itemId == NavigationTree.SHOW_ALL) {
+ // clear previous filters
+ getDataSource().removeAllContainerFilters();
+ showListView();
+ } else if (itemId == NavigationTree.SEARCH) {
+ showSearchView();
+ } else if (itemId instanceof SearchFilter) {
+ search((SearchFilter) itemId);
+ }
+ }
+ }
+ }
+
+ private void addNewContanct() {
+ showListView();
+ personForm.addContact();
+ }
+
+ public void search(SearchFilter searchFilter) {
+ // clear previous filters
+ getDataSource().removeAllContainerFilters();
+ // filter contacts with given filter
+ getDataSource().addContainerFilter(searchFilter.getPropertyId(),
+ searchFilter.getTerm(), true, false);
+ showListView();
+
+ getMainWindow().showNotification(
+ "Searched for " + searchFilter.getPropertyId() + "=*"
+ + searchFilter.getTerm() + "*, found "
+ + getDataSource().size() + " item(s).",
+ Notification.TYPE_TRAY_NOTIFICATION);
+ }
+
+ public void saveSearch(SearchFilter searchFilter) {
+ tree.addItem(searchFilter);
+ tree.setParent(searchFilter, NavigationTree.SEARCH);
+ // mark the saved search as a leaf (cannot have children)
+ tree.setChildrenAllowed(searchFilter, false);
+ // make sure "Search" is expanded
+ tree.expandItem(NavigationTree.SEARCH);
+ // select the saved search
+ tree.setValue(searchFilter);
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/data/Person.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/data/Person.java
new file mode 100644
index 0000000000..02f670f4f4
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/data/Person.java
@@ -0,0 +1,117 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.data;
+
+public class Person {
+ private String firstName = "";
+ private String lastName = "";
+ private String email = "";
+ private String phoneNumber = "";
+ private String streetAddress = "";
+ private Integer postalCode = null;
+ private String city = "";
+
+ /**
+ * @return the firstName
+ */
+ public String getFirstName() {
+ return firstName;
+ }
+
+ /**
+ * @param firstName
+ * the firstName to set
+ */
+ public void setFirstName(String firstName) {
+ this.firstName = firstName;
+ }
+
+ /**
+ * @return the lastName
+ */
+ public String getLastName() {
+ return lastName;
+ }
+
+ /**
+ * @param lastName
+ * the lastName to set
+ */
+ public void setLastName(String lastName) {
+ this.lastName = lastName;
+ }
+
+ /**
+ * @return the email
+ */
+ public String getEmail() {
+ return email;
+ }
+
+ /**
+ * @param email
+ * the email to set
+ */
+ public void setEmail(String email) {
+ this.email = email;
+ }
+
+ /**
+ * @return the phoneNumber
+ */
+ public String getPhoneNumber() {
+ return phoneNumber;
+ }
+
+ /**
+ * @param phoneNumber
+ * the phoneNumber to set
+ */
+ public void setPhoneNumber(String phoneNumber) {
+ this.phoneNumber = phoneNumber;
+ }
+
+ /**
+ * @return the streetAddress
+ */
+ public String getStreetAddress() {
+ return streetAddress;
+ }
+
+ /**
+ * @param streetAddress
+ * the streetAddress to set
+ */
+ public void setStreetAddress(String streetAddress) {
+ this.streetAddress = streetAddress;
+ }
+
+ /**
+ * @return the postalCode
+ */
+ public Integer getPostalCode() {
+ return postalCode;
+ }
+
+ /**
+ * @param postalCode
+ * the postalCode to set
+ */
+ public void setPostalCode(Integer postalCode) {
+ this.postalCode = postalCode;
+ }
+
+ /**
+ * @return the city
+ */
+ public String getCity() {
+ return city;
+ }
+
+ /**
+ * @param city
+ * the city to set
+ */
+ public void setCity(String city) {
+ this.city = city;
+ }
+
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/data/PersonContainer.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/data/PersonContainer.java
new file mode 100644
index 0000000000..b627f7e039
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/data/PersonContainer.java
@@ -0,0 +1,91 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.data;
+
+import java.util.Random;
+
+import com.itmill.toolkit.data.util.BeanItemContainer;
+
+public class PersonContainer extends BeanItemContainer<Person> {
+
+ /**
+ * Natural property order for Person bean. Used in tables and forms.
+ */
+ public static final Object[] NATURAL_COL_ORDER = new Object[] {
+ "firstName", "lastName", "email", "phoneNumber", "streetAddress",
+ "postalCode", "city" };
+
+ /**
+ * "Human readable" captions for properties in same order as in
+ * NATURAL_COL_ORDER.
+ */
+ public static final String[] COL_HEADERS_ENGLISH = new String[] {
+ "First name", "Last name", "Email", "Phone number",
+ "Street Address", "Postal Code", "City" };
+
+ public PersonContainer() throws InstantiationException,
+ IllegalAccessException {
+ super(Person.class);
+ }
+
+ public static PersonContainer createWithTestData() {
+ final String[] fnames = { "Peter", "Alice", "Joshua", "Mike", "Olivia",
+ "Nina", "Alex", "Rita", "Dan", "Umberto", "Henrik", "Rene",
+ "Lisa", "Marge" };
+ final String[] lnames = { "Smith", "Gordon", "Simpson", "Brown",
+ "Clavel", "Simons", "Verne", "Scott", "Allison", "Gates",
+ "Rowling", "Barks", "Ross", "Schneider", "Tate" };
+ final String cities[] = { "Amsterdam", "Berlin", "Helsinki",
+ "Hong Kong", "London", "Luxemburg", "New York", "Oslo",
+ "Paris", "Rome", "Stockholm", "Tokyo", "Turku" };
+ final String streets[] = { "4215 Blandit Av.", "452-8121 Sem Ave",
+ "279-4475 Tellus Road", "4062 Libero. Av.", "7081 Pede. Ave",
+ "6800 Aliquet St.", "P.O. Box 298, 9401 Mauris St.",
+ "161-7279 Augue Ave", "P.O. Box 496, 1390 Sagittis. Rd.",
+ "448-8295 Mi Avenue", "6419 Non Av.",
+ "659-2538 Elementum Street", "2205 Quis St.",
+ "252-5213 Tincidunt St.", "P.O. Box 175, 4049 Adipiscing Rd.",
+ "3217 Nam Ave", "P.O. Box 859, 7661 Auctor St.",
+ "2873 Nonummy Av.", "7342 Mi, Avenue",
+ "539-3914 Dignissim. Rd.", "539-3675 Magna Avenue",
+ "Ap #357-5640 Pharetra Avenue", "416-2983 Posuere Rd.",
+ "141-1287 Adipiscing Avenue", "Ap #781-3145 Gravida St.",
+ "6897 Suscipit Rd.", "8336 Purus Avenue", "2603 Bibendum. Av.",
+ "2870 Vestibulum St.", "Ap #722 Aenean Avenue",
+ "446-968 Augue Ave", "1141 Ultricies Street",
+ "Ap #992-5769 Nunc Street", "6690 Porttitor Avenue",
+ "Ap #105-1700 Risus Street",
+ "P.O. Box 532, 3225 Lacus. Avenue", "736 Metus Street",
+ "414-1417 Fringilla Street", "Ap #183-928 Scelerisque Road",
+ "561-9262 Iaculis Avenue" };
+ PersonContainer c = null;
+ Random r = new Random(0);
+ try {
+ c = new PersonContainer();
+ for (int i = 0; i < 100; i++) {
+ Person p = new Person();
+ p.setFirstName(fnames[r.nextInt(fnames.length)]);
+ p.setLastName(lnames[r.nextInt(lnames.length)]);
+ p.setCity(cities[r.nextInt(cities.length)]);
+ p.setEmail(p.getFirstName().toLowerCase() + "."
+ + p.getLastName().toLowerCase() + "@itmill.com");
+ p.setPhoneNumber("+358 02 555 " + r.nextInt(10) + r.nextInt(10)
+ + r.nextInt(10) + r.nextInt(10));
+ int n = r.nextInt(100000);
+ if (n < 10000) {
+ n += 10000;
+ }
+ p.setPostalCode(n);
+ p.setStreetAddress(streets[r.nextInt(streets.length)]);
+ c.addItem(p);
+ }
+ } catch (InstantiationException e) {
+ // TODO Auto-generated catch block
+ e.printStackTrace();
+ } catch (IllegalAccessException e) {
+ // TODO Auto-generated catch block
+ e.printStackTrace();
+ }
+
+ return c;
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/data/SearchFilter.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/data/SearchFilter.java
new file mode 100644
index 0000000000..137917a341
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/data/SearchFilter.java
@@ -0,0 +1,41 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.data;
+
+public class SearchFilter {
+
+ private final String term;
+ private final Object propertyId;
+ private String searchName;
+
+ public SearchFilter(Object propertyId, String searchTerm, String name) {
+ this.propertyId = propertyId;
+ this.term = searchTerm;
+ this.searchName = name;
+ }
+
+ /**
+ * @return the term
+ */
+ public String getTerm() {
+ return term;
+ }
+
+ /**
+ * @return the propertyId
+ */
+ public Object getPropertyId() {
+ return propertyId;
+ }
+
+ /**
+ * @return the name of the search
+ */
+ public String getSearchName() {
+ return searchName;
+ }
+
+ @Override
+ public String toString() {
+ return getSearchName();
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/HelpWindow.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/HelpWindow.java
new file mode 100644
index 0000000000..d5c08721d4
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/HelpWindow.java
@@ -0,0 +1,17 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import com.itmill.toolkit.ui.Label;
+import com.itmill.toolkit.ui.Window;
+
+public class HelpWindow extends Window {
+ private static final String HELP_HTML_SNIPPET = "This is "
+ + "an application built during <strong><a href=\""
+ + "http://dev.itmill.com/\">Toolkit</a></strong> "
+ + "tutorial. Hopefully it don't need any real help.";
+
+ public HelpWindow() {
+ setCaption("Address Book help");
+ addComponent(new Label(HELP_HTML_SNIPPET, Label.CONTENT_XHTML));
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/ListView.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/ListView.java
new file mode 100644
index 0000000000..4f5425d39d
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/ListView.java
@@ -0,0 +1,12 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import com.itmill.toolkit.ui.SplitPanel;
+
+public class ListView extends SplitPanel {
+ public ListView(PersonList personList, PersonForm personForm) {
+ addStyleName("view");
+ setFirstComponent(personList);
+ setSecondComponent(personForm);
+ setSplitPosition(40);
+ }
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/NavigationTree.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/NavigationTree.java
new file mode 100644
index 0000000000..c44d27b9ca
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/NavigationTree.java
@@ -0,0 +1,28 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import com.itmill.toolkit.demo.tutorial.addressbook.AddressBookApplication;
+import com.itmill.toolkit.event.ItemClickEvent.ItemClickListener;
+import com.itmill.toolkit.ui.Tree;
+
+public class NavigationTree extends Tree {
+ public static final Object SHOW_ALL = "Show all";
+ public static final Object SEARCH = "Search";
+
+ public NavigationTree(AddressBookApplication app) {
+ addItem(SHOW_ALL);
+ addItem(SEARCH);
+
+ setChildrenAllowed(SHOW_ALL, false);
+
+ /*
+ * We want items to be selectable but do not want the user to be able to
+ * de-select an item.
+ */
+ setSelectable(true);
+ setNullSelectionAllowed(false);
+
+ // Make application handle item click events
+ addListener((ItemClickListener) app);
+
+ }
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonForm.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonForm.java
new file mode 100644
index 0000000000..f7f9e0aa77
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonForm.java
@@ -0,0 +1,176 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import java.util.Arrays;
+import java.util.Iterator;
+import java.util.List;
+
+import com.itmill.toolkit.data.Item;
+import com.itmill.toolkit.data.util.BeanItem;
+import com.itmill.toolkit.demo.tutorial.addressbook.AddressBookApplication;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.Person;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.PersonContainer;
+import com.itmill.toolkit.demo.tutorial.addressbook.validators.EmailValidator;
+import com.itmill.toolkit.demo.tutorial.addressbook.validators.PostalCodeValidator;
+import com.itmill.toolkit.ui.BaseFieldFactory;
+import com.itmill.toolkit.ui.Button;
+import com.itmill.toolkit.ui.ComboBox;
+import com.itmill.toolkit.ui.Component;
+import com.itmill.toolkit.ui.Field;
+import com.itmill.toolkit.ui.Form;
+import com.itmill.toolkit.ui.HorizontalLayout;
+import com.itmill.toolkit.ui.TextField;
+import com.itmill.toolkit.ui.Button.ClickEvent;
+import com.itmill.toolkit.ui.Button.ClickListener;
+
+public class PersonForm extends Form implements ClickListener {
+
+ private Button save = new Button("Save", (ClickListener) this);
+ private Button cancel = new Button("Cancel", (ClickListener) this);
+ private Button edit = new Button("Edit", (ClickListener) this);
+ private final ComboBox cities = new ComboBox("City");
+
+ private AddressBookApplication app;
+ private boolean newContactMode = false;
+ private Person newPerson = null;
+
+ public PersonForm(AddressBookApplication app) {
+ this.app = app;
+
+ /*
+ * Enable buffering so that commit() must be called for the form before
+ * input is written to the data. (Form input is not written immediately
+ * through to the underlying object.)
+ */
+ setWriteThrough(false);
+
+ HorizontalLayout footer = new HorizontalLayout();
+ footer.setSpacing(true);
+ footer.addComponent(save);
+ footer.addComponent(cancel);
+ footer.addComponent(edit);
+ footer.setVisible(false);
+
+ setFooter(footer);
+
+ /* Allow the user to enter new cities */
+ cities.setNewItemsAllowed(true);
+ /* We do not want to use null values */
+ cities.setNullSelectionAllowed(false);
+ /* Add an empty city used for selecting no city */
+ cities.addItem("");
+
+ /* Populate cities select using the cities in the data container */
+ PersonContainer ds = app.getDataSource();
+ for (Iterator<Person> it = ds.getItemIds().iterator(); it.hasNext();) {
+ String city = (it.next()).getCity();
+ cities.addItem(city);
+ }
+
+ /*
+ * Field factory for overriding how the component for city selection is
+ * created
+ */
+ setFieldFactory(new BaseFieldFactory() {
+ @Override
+ public Field createField(Item item, Object propertyId,
+ Component uiContext) {
+ if (propertyId.equals("city")) {
+ cities.setWidth("200px");
+ return cities;
+ }
+
+ Field field = super.createField(item, propertyId, uiContext);
+ if (propertyId.equals("postalCode")) {
+ TextField tf = (TextField) field;
+ /*
+ * We do not want to display "null" to the user when the
+ * field is empty
+ */
+ tf.setNullRepresentation("");
+
+ /* Add a validator for postalCode and make it required */
+ tf.addValidator(new PostalCodeValidator());
+ tf.setRequired(true);
+ } else if (propertyId.equals("email")) {
+ /* Add a validator for email and make it required */
+ field.addValidator(new EmailValidator());
+ field.setRequired(true);
+
+ }
+
+ field.setWidth("200px");
+ return field;
+ }
+ });
+ }
+
+ public void buttonClick(ClickEvent event) {
+ Button source = event.getButton();
+
+ if (source == save) {
+ /* If the given input is not valid there is no point in continuing */
+ if (!isValid()) {
+ return;
+ }
+
+ commit();
+ if (newContactMode) {
+ /* We need to add the new person to the container */
+ Item addedItem = app.getDataSource().addItem(newPerson);
+ /*
+ * We must update the form to use the Item from our datasource
+ * as we are now in edit mode (no longer in add mode)
+ */
+ setItemDataSource(addedItem);
+
+ newContactMode = false;
+ }
+ setReadOnly(true);
+ } else if (source == cancel) {
+ if (newContactMode) {
+ newContactMode = false;
+ /* Clear the form and make it invisible */
+ setItemDataSource(null);
+ } else {
+ discard();
+ }
+ setReadOnly(true);
+ } else if (source == edit) {
+ setReadOnly(false);
+ }
+ }
+
+ @Override
+ public void setItemDataSource(Item newDataSource) {
+ newContactMode = false;
+
+ if (newDataSource != null) {
+ List<Object> orderedProperties = Arrays
+ .asList(PersonContainer.NATURAL_COL_ORDER);
+ super.setItemDataSource(newDataSource, orderedProperties);
+
+ setReadOnly(true);
+ getFooter().setVisible(true);
+ } else {
+ super.setItemDataSource(null);
+ getFooter().setVisible(false);
+ }
+ }
+
+ @Override
+ public void setReadOnly(boolean readOnly) {
+ super.setReadOnly(readOnly);
+ save.setVisible(!readOnly);
+ cancel.setVisible(!readOnly);
+ edit.setVisible(readOnly);
+ }
+
+ public void addContact() {
+ // Create a temporary item for the form
+ newPerson = new Person();
+ setItemDataSource(new BeanItem(newPerson));
+ newContactMode = true;
+ setReadOnly(false);
+ }
+
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonList.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonList.java
new file mode 100644
index 0000000000..12007683be
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/PersonList.java
@@ -0,0 +1,45 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import com.itmill.toolkit.demo.tutorial.addressbook.AddressBookApplication;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.Person;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.PersonContainer;
+import com.itmill.toolkit.terminal.ExternalResource;
+import com.itmill.toolkit.ui.Component;
+import com.itmill.toolkit.ui.Link;
+import com.itmill.toolkit.ui.Table;
+
+public class PersonList extends Table {
+ public PersonList(AddressBookApplication app) {
+ setSizeFull();
+ setContainerDataSource(app.getDataSource());
+
+ setVisibleColumns(PersonContainer.NATURAL_COL_ORDER);
+ setColumnHeaders(PersonContainer.COL_HEADERS_ENGLISH);
+
+ setColumnCollapsingAllowed(true);
+ setColumnReorderingAllowed(true);
+
+ /*
+ * Make table selectable, react immediatedly to user events, and pass
+ * events to the controller (our main application)
+ */
+ setSelectable(true);
+ setImmediate(true);
+ addListener((ValueChangeListener) app);
+ /* We don't want to allow users to de-select a row */
+ setNullSelectionAllowed(false);
+
+ // customize email column to have mailto: links using column generator
+ addGeneratedColumn("email", new ColumnGenerator() {
+ public Component generateCell(Table source, Object itemId,
+ Object columnId) {
+ Person p = (Person) itemId;
+ Link l = new Link();
+ l.setResource(new ExternalResource("mailto:" + p.getEmail()));
+ l.setCaption(p.getEmail());
+ return l;
+ }
+ });
+ }
+
+} \ No newline at end of file
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SearchView.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SearchView.java
new file mode 100644
index 0000000000..4086c95d73
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SearchView.java
@@ -0,0 +1,99 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import com.itmill.toolkit.demo.tutorial.addressbook.AddressBookApplication;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.PersonContainer;
+import com.itmill.toolkit.demo.tutorial.addressbook.data.SearchFilter;
+import com.itmill.toolkit.ui.Button;
+import com.itmill.toolkit.ui.CheckBox;
+import com.itmill.toolkit.ui.FormLayout;
+import com.itmill.toolkit.ui.NativeSelect;
+import com.itmill.toolkit.ui.Panel;
+import com.itmill.toolkit.ui.TextField;
+import com.itmill.toolkit.ui.Button.ClickEvent;
+import com.itmill.toolkit.ui.Button.ClickListener;
+import com.itmill.toolkit.ui.Window.Notification;
+
+public class SearchView extends Panel {
+
+ private TextField tf;
+ private NativeSelect fieldToSearch;
+ private CheckBox saveSearch;
+ private TextField searchName;
+ private AddressBookApplication app;
+
+ public SearchView(final AddressBookApplication app) {
+ this.app = app;
+ addStyleName("view");
+
+ setCaption("Search contacts");
+ setSizeFull();
+
+ /* Use a FormLayout as main layout for this Panel */
+ FormLayout formLayout = new FormLayout();
+ setLayout(formLayout);
+
+ /* Create UI components */
+ tf = new TextField("Search term");
+ fieldToSearch = new NativeSelect("Field to search");
+ saveSearch = new CheckBox("Save search");
+ searchName = new TextField("Search name");
+ Button search = new Button("Search");
+
+ /* Initialize fieldToSearch */
+ for (int i = 0; i < PersonContainer.NATURAL_COL_ORDER.length; i++) {
+ fieldToSearch.addItem(PersonContainer.NATURAL_COL_ORDER[i]);
+ fieldToSearch.setItemCaption(PersonContainer.NATURAL_COL_ORDER[i],
+ PersonContainer.COL_HEADERS_ENGLISH[i]);
+ }
+
+ fieldToSearch.setValue("lastName");
+ fieldToSearch.setNullSelectionAllowed(false);
+
+ /* Initialize save checkbox */
+ saveSearch.setValue(true);
+ saveSearch.setImmediate(true);
+ saveSearch.addListener(new ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ searchName.setVisible(event.getButton().booleanValue());
+ }
+ });
+
+ search.addListener(new Button.ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ performSearch();
+ }
+
+ });
+
+ /* Add all the created components to the form */
+ addComponent(tf);
+ addComponent(fieldToSearch);
+ addComponent(saveSearch);
+ addComponent(searchName);
+ addComponent(search);
+ }
+
+ private void performSearch() {
+ String searchTerm = (String) tf.getValue();
+ if (searchTerm == null || searchTerm.equals("")) {
+ getWindow().showNotification("Search term cannot be empty!",
+ Notification.TYPE_WARNING_MESSAGE);
+ return;
+ }
+
+ SearchFilter searchFilter = new SearchFilter(fieldToSearch.getValue(),
+ searchTerm, (String) searchName.getValue());
+ if (saveSearch.booleanValue()) {
+ if (searchName.getValue() == null
+ || searchName.getValue().equals("")) {
+ getWindow().showNotification(
+ "Please enter a name for your search!",
+ Notification.TYPE_WARNING_MESSAGE);
+ return;
+ }
+ app.saveSearch(searchFilter);
+ }
+ app.search(searchFilter);
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SharingOptions.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SharingOptions.java
new file mode 100644
index 0000000000..772247cdfb
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/ui/SharingOptions.java
@@ -0,0 +1,39 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.ui;
+
+import com.itmill.toolkit.ui.Button;
+import com.itmill.toolkit.ui.CheckBox;
+import com.itmill.toolkit.ui.Label;
+import com.itmill.toolkit.ui.Window;
+import com.itmill.toolkit.ui.Button.ClickEvent;
+
+public class SharingOptions extends Window {
+ public SharingOptions() {
+ /*
+ * Make the window modal, which will disable all other components while
+ * it is visible
+ */
+ setModal(true);
+
+ /* Make the sub window 50% the size of the browser window */
+ setWidth("50%");
+ /*
+ * Center the window both horizontally and vertically in the browser
+ * window
+ */
+ center();
+
+ setCaption("Sharing options");
+ addComponent(new Label(
+ "With these setting you can modify contact sharing "
+ + "options. (non-functional, example of modal dialog)"));
+ addComponent(new CheckBox("Gmail"));
+ addComponent(new CheckBox(".Mac"));
+ Button close = new Button("OK");
+ close.addListener(new Button.ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ SharingOptions.this.close();
+ }
+ });
+ addComponent(close);
+ }
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/validators/EmailValidator.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/validators/EmailValidator.java
new file mode 100644
index 0000000000..609a216f95
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/validators/EmailValidator.java
@@ -0,0 +1,22 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.validators;
+
+import com.itmill.toolkit.data.Validator;
+
+public class EmailValidator implements Validator {
+
+ public boolean isValid(Object value) {
+ if (value == null || !(value instanceof String)) {
+ return false;
+ }
+
+ return ((String) value).matches(".+@.+\\..+");
+ }
+
+ public void validate(Object value) throws InvalidValueException {
+ if (!isValid(value)) {
+ throw new InvalidValueException(
+ "Email must contain '@' and have full domain.");
+ }
+ }
+
+}
diff --git a/src/com/itmill/toolkit/demo/tutorial/addressbook/validators/PostalCodeValidator.java b/src/com/itmill/toolkit/demo/tutorial/addressbook/validators/PostalCodeValidator.java
new file mode 100644
index 0000000000..5558a0d1dd
--- /dev/null
+++ b/src/com/itmill/toolkit/demo/tutorial/addressbook/validators/PostalCodeValidator.java
@@ -0,0 +1,22 @@
+package com.itmill.toolkit.demo.tutorial.addressbook.validators;
+
+import com.itmill.toolkit.data.Validator;
+
+public class PostalCodeValidator implements Validator {
+
+ public boolean isValid(Object value) {
+ if (value == null || !(value instanceof String)) {
+ return false;
+ }
+
+ return ((String) value).matches("[1-9][0-9]{4}");
+ }
+
+ public void validate(Object value) throws InvalidValueException {
+ if (!isValid(value)) {
+ throw new InvalidValueException(
+ "Postal code must be a five digit number and cannot start with a zero.");
+ }
+ }
+
+} \ No newline at end of file