diff --git a/examples/notebooks/widgets/Tutorials - Alignment.ipynb b/examples/notebooks/widgets/Tutorials - Alignment.ipynb
new file mode 100644
index 0000000..6ceb400
--- /dev/null
+++ b/examples/notebooks/widgets/Tutorials - Alignment.ipynb
@@ -0,0 +1,277 @@
+{
+ "metadata": {
+ "name": ""
+ },
+ "nbformat": 3,
+ "nbformat_minor": 0,
+ "worksheets": [
+ {
+ "cells": [
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "from IPython.html import widgets # Widget definitions\n",
+ "from IPython.display import display # Used to display widgets in the notebook\n",
+ "\n",
+ "# Enable widgets in this notebook\n",
+ "widgets.init_widget_js()"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/bool.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/int_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/int.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/selection.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/string.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/float.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/container.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/multicontainer.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/button.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/float_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ }
+ ],
+ "prompt_number": 1
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Alignment"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Most widgets have a `description` property which allows a label for the widget to be defined. The label of the widget has a fixed minimum width. The text of the label is always right aligned and the widget is left aligned (as seen below) "
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "display(widgets.StringWidget(description=\"a:\"))\n",
+ "display(widgets.StringWidget(description=\"aa:\"))\n",
+ "display(widgets.StringWidget(description=\"aaa:\"))"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 2
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "If a label is longer than the minimum width, the widget is shifted to the right (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "display(widgets.StringWidget(description=\"a:\"))\n",
+ "display(widgets.StringWidget(description=\"aa:\"))\n",
+ "display(widgets.StringWidget(description=\"aaa:\"))\n",
+ "display(widgets.StringWidget(description=\"aaaaaaaaaaaaaaaaaa:\"))"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 3
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "If a `description` is not set for the widget, the label is not displayed (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "display(widgets.StringWidget(description=\"a:\"))\n",
+ "display(widgets.StringWidget(description=\"aa:\"))\n",
+ "display(widgets.StringWidget(description=\"aaa:\"))\n",
+ "display(widgets.StringWidget())"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 4
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Custom Alignment"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "`ContainerWidget`s allow for custom alignment of widgets. The `hbox` and `vbox` methods (parameterless) cause the `ContainerWidget` to both horizontally and vertically align its children. The following example compares `vbox` to `hbox`."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "child_style = {\n",
+ " 'background': '#77CC77',\n",
+ " 'padding': '25px',\n",
+ " 'margin': '5px',\n",
+ " 'font-size': 'xx-large',\n",
+ " 'color': 'white',\n",
+ "}\n",
+ "\n",
+ "def make_container(title):\n",
+ " display(widgets.StringWidget(default_view_name='LabelView', value='
' + title + '
'))\n",
+ " container = widgets.ContainerWidget()\n",
+ " container.set_css('background', '#999999')\n",
+ " display(container)\n",
+ " return container\n",
+ "\n",
+ "def fill_container(container):\n",
+ " components = []\n",
+ " for i in range(3):\n",
+ " components.append(widgets.StringWidget(parent=container, default_view_name='LabelView', value=\"ABC\"[i]))\n",
+ " components[i].set_css(child_style)\n",
+ " display(components[i])\n",
+ " \n",
+ "container = make_container('VBox')\n",
+ "container.vbox()\n",
+ "fill_container(container)\n",
+ "\n",
+ "container = make_container('HBox')\n",
+ "container.hbox()\n",
+ "fill_container(container)\n"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 5
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The `ContainerWidget` also supports `start`, `center`, and `end` methods (parameterless) that adjust the alignment of the widgets on the axis that they are being rendered on. Below is an example of the different alignments."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = make_container('HBox Start')\n",
+ "container.hbox()\n",
+ "container.start()\n",
+ "fill_container(container)\n",
+ " \n",
+ "container = make_container('HBox Center')\n",
+ "container.hbox()\n",
+ "container.center()\n",
+ "fill_container(container)\n",
+ " \n",
+ "container = make_container('HBox End')\n",
+ "container.hbox()\n",
+ "container.end()\n",
+ "fill_container(container)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 6
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "By default the widget area is a `vbox`; however, there are many uses for a `hbox`. The example below uses a `hbox` to display a set of vertical sliders, like an equalizer."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.ContainerWidget()\n",
+ "container.hbox()\n",
+ "for i in range(15):\n",
+ " widgets.FloatRangeWidget(orientation='vertical', parent=container, description=str(i+1), value=50.0)\n",
+ "display(container)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 7
+ }
+ ],
+ "metadata": {}
+ }
+ ]
+}
\ No newline at end of file
diff --git a/examples/notebooks/widgets/Tutorials - Basics.ipynb b/examples/notebooks/widgets/Tutorials - Basics.ipynb
new file mode 100644
index 0000000..fc4e4f4
--- /dev/null
+++ b/examples/notebooks/widgets/Tutorials - Basics.ipynb
@@ -0,0 +1,391 @@
+{
+ "metadata": {
+ "name": ""
+ },
+ "nbformat": 3,
+ "nbformat_minor": 0,
+ "worksheets": [
+ {
+ "cells": [
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "To enable the use IPython widgets in the notebook, the widget namespace and display function need to be imported. The Javascript dependencies need to be loaded via `IPython.html.widgets.init_widget_js()`. This method needs to be called each time the notebook webpage is refreshed."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "from IPython.html import widgets # Widget definitions\n",
+ "from IPython.display import display # Used to display widgets in the notebook\n",
+ "\n",
+ "# Enable widgets in this notebook\n",
+ "widgets.init_widget_js()"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/bool.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/int_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/int.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/selection.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/string.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/float.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/container.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/multicontainer.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/button.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/float_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ }
+ ],
+ "prompt_number": 1
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Basic Widgets"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The IPython notebook comes preloaded with basic widgets that represent common data types. These widgets are\n",
+ "\n",
+ "- BoolWidget : boolean \n",
+ "- FloatRangeWidget : bounded float \n",
+ "- FloatWidget : unbounded float \n",
+ "- IntRangeWidget : bounded integer \n",
+ "- IntWidget : unbounded integer \n",
+ "- SelectionWidget : enumeration \n",
+ "- StringWidget : string \n",
+ "\n",
+ "A few special widgets are also included, that can be used to capture events and change how other widgets are displayed. These widgets are\n",
+ "\n",
+ "- ButtonWidget \n",
+ "- ContainerWidget \n",
+ "- MulticontainerWidget \n",
+ "\n",
+ "To see the complete list of widgets, one can execute the following"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "[widget for widget in dir(widgets) if widget.endswith('Widget')]"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "metadata": {},
+ "output_type": "pyout",
+ "prompt_number": 2,
+ "text": [
+ "['BoolWidget',\n",
+ " 'ButtonWidget',\n",
+ " 'ContainerWidget',\n",
+ " 'FloatRangeWidget',\n",
+ " 'FloatWidget',\n",
+ " 'IntRangeWidget',\n",
+ " 'IntWidget',\n",
+ " 'MulticontainerWidget',\n",
+ " 'SelectionWidget',\n",
+ " 'StringWidget',\n",
+ " 'Widget']"
+ ]
+ }
+ ],
+ "prompt_number": 2
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The basic widgets can all be constructed without arguments. The following creates a FloatRangeWidget without displaying it"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "mywidget = widgets.FloatRangeWidget()"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 3
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Constructing a widget does not display it on the page. To display a widget, the widget must be passed to the IPython `display(object)` method. `mywidget` is displayed by"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "display(mywidget)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 4
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "It's important to realize that widgets are not the same as output, even though they are displayed with `display`. Widgets are drawn in a special widget area. That area is marked with a close button which allows you to collapse the widgets. Widgets cannot be interleaved with output. Doing so would break the ability to make simple animations using `clear_output`.\n",
+ "\n",
+ "Widgets are manipulated via special instance properties (traitlets). The names of these instance properties are listed in the widget's `keys` property (as seen below). A few of these properties are common to most, if not all, widgets. The common properties are `value`, `description`, `visible`, and `disabled`. `_css`, `_add_class`, and `_remove_class` are internal properties that exist in all widgets and should not be modified."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "mywidget.keys"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "metadata": {},
+ "output_type": "pyout",
+ "prompt_number": 5,
+ "text": [
+ "['visible',\n",
+ " '_css',\n",
+ " '_add_class',\n",
+ " '_remove_class',\n",
+ " 'value',\n",
+ " 'step',\n",
+ " 'max',\n",
+ " 'min',\n",
+ " 'disabled',\n",
+ " 'orientation',\n",
+ " 'description']"
+ ]
+ }
+ ],
+ "prompt_number": 5
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Changing a widget's property value will automatically update that widget everywhere it is displayed in the notebook. Here the value of `mywidget` is set. The slider shown above (after input 4) updates automatically to the new value. In reverse, changing the value of the displayed widget will update the property's value."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "mywidget.value = 25.0"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 6
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "After changing the widget's value in the notebook by hand to 0.0 (sliding the bar to the far left)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "mywidget.value"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "metadata": {},
+ "output_type": "pyout",
+ "prompt_number": 7,
+ "text": [
+ "0.0"
+ ]
+ }
+ ],
+ "prompt_number": 7
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Widget property values can also be set with kwargs during the construction of the widget (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "mysecondwidget = widgets.SelectionWidget(values=[\"Item A\", \"Item B\", \"Item C\"], value=\"Nothing Selected\")\n",
+ "display(mysecondwidget)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 8
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Views"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The data types that most of the widgets represent can be displayed more than one way. A `view` is a visual representation of a widget in the notebook. In the example in the section above, the default `view` for the `FloatRangeWidget` is used. The default view is set in the widgets `default_view_name` instance property (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "mywidget.default_view_name"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "metadata": {},
+ "output_type": "pyout",
+ "prompt_number": 9,
+ "text": [
+ "u'FloatSliderView'"
+ ]
+ }
+ ],
+ "prompt_number": 9
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "When a widget is displayed using `display(...)`, the `default_view_name` is used to determine what view type should be used to display the widget. View names are case sensitive. Sometimes the default view isn't the best view to represent a piece of data. To change what view is used, either the `default_view_name` can be changed or the `view_name` kwarg of `display` can be set. This also can be used to display one widget multiple ways in one output (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "display(mywidget)\n",
+ "display(mywidget, view_name=\"FloatTextView\")"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 10
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Some views work with multiple different widget types and some views only work with one. The complete list of views and supported widgets is below. The default views are italicized.\n",
+ "\n",
+ "| Widget Name | View Names |\n",
+ "|:-----------------------|:--------------------|\n",
+ "| BoolWidget | *CheckboxView* |\n",
+ "| | ToggleButtonView |\n",
+ "| ButtonWidget | *ButtonView* |\n",
+ "| ContainerWidget | *ContainerView* |\n",
+ "| FloatRangeWidget | *FloatSliderView* |\n",
+ "| | FloatTextView |\n",
+ "| | ProgressView |\n",
+ "| FloatWidget | *FloatTextView* |\n",
+ "| IntRangeWidget | *IntSliderView* |\n",
+ "| | IntTextView |\n",
+ "| | ProgressView |\n",
+ "| IntWidget | *IntTextView* |\n",
+ "| MulticontainerWidget | AccordionView |\n",
+ "| | *TabView* |\n",
+ "| SelectionWidget | ToggleButtonsView |\n",
+ "| | RadioButtonsView |\n",
+ "| | *DropdownView* |\n",
+ "| StringWidget | LabelView |\n",
+ "| | TextAreaView |\n",
+ "| | *TextBoxView* |\n"
+ ]
+ }
+ ],
+ "metadata": {}
+ }
+ ]
+}
\ No newline at end of file
diff --git a/examples/notebooks/widgets/Tutorials - Events.ipynb b/examples/notebooks/widgets/Tutorials - Events.ipynb
new file mode 100644
index 0000000..cee42a0
--- /dev/null
+++ b/examples/notebooks/widgets/Tutorials - Events.ipynb
@@ -0,0 +1,310 @@
+{
+ "metadata": {
+ "name": ""
+ },
+ "nbformat": 3,
+ "nbformat_minor": 0,
+ "worksheets": [
+ {
+ "cells": [
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "from IPython.html import widgets # Widget definitions\n",
+ "from IPython.display import display # Used to display widgets in the notebook\n",
+ "\n",
+ "# Enable widgets in this notebook\n",
+ "widgets.init_widget_js()"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/bool.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/int_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/int.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/selection.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/string.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/float.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/container.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/multicontainer.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/button.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/float_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ }
+ ],
+ "prompt_number": 1
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Traitlet Events"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The widget properties are IPython traitlets. Traitlets are eventful. To handle property value changes, the `on_trait_change` method of the widget can be used to register an event handling callback. The doc string for `on_trait_change` can be seen below. Both the `name` and `remove` properties are optional."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "print(widgets.Widget.on_trait_change.__doc__)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Setup a handler to be called when a trait changes.\n",
+ "\n",
+ " This is used to setup dynamic notifications of trait changes.\n",
+ "\n",
+ " Static handlers can be created by creating methods on a HasTraits\n",
+ " subclass with the naming convention '_[traitname]_changed'. Thus,\n",
+ " to create static handler for the trait 'a', create the method\n",
+ " _a_changed(self, name, old, new) (fewer arguments can be used, see\n",
+ " below).\n",
+ "\n",
+ " Parameters\n",
+ " ----------\n",
+ " handler : callable\n",
+ " A callable that is called when a trait changes. Its\n",
+ " signature can be handler(), handler(name), handler(name, new)\n",
+ " or handler(name, old, new).\n",
+ " name : list, str, None\n",
+ " If None, the handler will apply to all traits. If a list\n",
+ " of str, handler will apply to all names in the list. If a\n",
+ " str, the handler will apply just to that name.\n",
+ " remove : bool\n",
+ " If False (the default), then install the handler. If True\n",
+ " then unintall it.\n",
+ " \n"
+ ]
+ }
+ ],
+ "prompt_number": 2
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Mentioned in the doc string, the callback registered can have 4 possible signatures:\n",
+ "\n",
+ "- callback()\n",
+ "- callback(trait_name)\n",
+ "- callback(trait_name, new_value)\n",
+ "- callback(trait_name, old_value, new_value)\n",
+ "\n",
+ "An example of how to output an IntRangeWiget's value as it is changed can be seen below."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "intrange = widgets.IntRangeWidget()\n",
+ "display(intrange)\n",
+ "\n",
+ "def on_value_change(name, value):\n",
+ " print value\n",
+ "\n",
+ "intrange.on_trait_change(on_value_change, 'value')"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 3
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Specialized Events"
+ ]
+ },
+ {
+ "cell_type": "heading",
+ "level": 2,
+ "metadata": {},
+ "source": [
+ "Button On Click Event"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The `ButtonWidget` is a special widget, like the `ContainerWidget` and `MulticontainerWidget`, that isn't used to represent a data type. Instead the button widget is used to handle mouse clicks. The `on_click` method of the `ButtonWidget` can be used to register a click even handler. The doc string of the `on_click` can be seen below."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "print(widgets.ButtonWidget.on_click.__doc__)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Register a callback to execute when the button is clicked. The\n",
+ " callback can either accept no parameters or one sender parameter:\n",
+ " - callback()\n",
+ " - callback(sender)\n",
+ " If the callback has a sender parameter, the ButtonWidget instance that\n",
+ " called the callback will be passed into the method as the sender.\n",
+ "\n",
+ " Parameters\n",
+ " ----------\n",
+ " remove : bool (optional)\n",
+ " Set to true to remove the callback from the list of callbacks.\n"
+ ]
+ }
+ ],
+ "prompt_number": 4
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Button clicks are tracked by the `clicks` property of the button widget. By using the `on_click` method and the `clicks` property, a button that outputs how many times it has been clicked is shown below."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "button = widgets.ButtonWidget(description=\"Click Me!\")\n",
+ "display(button)\n",
+ "\n",
+ "def on_button_clicked(sender):\n",
+ " print(\"Button clicked %d times.\" % sender.clicks)\n",
+ "\n",
+ "button.on_click(on_button_clicked)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Button clicked 1 times.\n"
+ ]
+ },
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Button clicked 2 times.\n"
+ ]
+ },
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Button clicked 3 times.\n"
+ ]
+ }
+ ],
+ "prompt_number": 5
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Event handlers can also be used to create widgets. In the example below, clicking a button spawns another button with a description equal to how many times the parent button had been clicked at the time."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "def show_button(sender=None):\n",
+ " button = widgets.ButtonWidget()\n",
+ " button.description = \"%d\" % (sender.clicks if sender is not None else 0)\n",
+ " display(button)\n",
+ " button.on_click(show_button)\n",
+ "show_button()\n",
+ " "
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 6
+ }
+ ],
+ "metadata": {}
+ }
+ ]
+}
\ No newline at end of file
diff --git a/examples/notebooks/widgets/Tutorials - Placement.ipynb b/examples/notebooks/widgets/Tutorials - Placement.ipynb
new file mode 100644
index 0000000..2dc2a57
--- /dev/null
+++ b/examples/notebooks/widgets/Tutorials - Placement.ipynb
@@ -0,0 +1,326 @@
+{
+ "metadata": {
+ "name": ""
+ },
+ "nbformat": 3,
+ "nbformat_minor": 0,
+ "worksheets": [
+ {
+ "cells": [
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "from IPython.html import widgets # Widget definitions\n",
+ "from IPython.display import display # Used to display widgets in the notebook\n",
+ "\n",
+ "# Enable widgets in this notebook\n",
+ "widgets.init_widget_js()"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/bool.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/int_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/int.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/selection.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/string.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/float.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/container.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/multicontainer.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/button.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"/static/notebook/js/widgets/float_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ }
+ ],
+ "prompt_number": 1
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Parent/Child Relationships"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "To display widget A inside widget B, widget A must be a child of widget B. With IPython widgets, the widgets are instances that live in the back-end (usally Python). There can be multiple views displayed in the front-end that represent one widget in the backend. Each view can be displayed at a different time, or even displayed two or more times in the same output. Because of this, the parent of a widget can only be set before the widget has been displayed.\n",
+ "\n",
+ "Every widget has a `parent` property. This property can be set via a kwarg in the widget's constructor or after construction, but before display. Calling display on an object with children automatically displays those children too (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.MulticontainerWidget()\n",
+ "\n",
+ "floatrange = widgets.FloatRangeWidget(parent=container) # You can set the parent in the constructor,\n",
+ "\n",
+ "string = widgets.StringWidget()\n",
+ "string.parent = container # or after the widget has been created.\n",
+ "\n",
+ "display(container) # Displays the `container` and all of it's children."
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 2
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Children can also be added to parents after the parent has been displayed. If the children are added after the parent has already been displayed, the children must be displayed themselves.\n",
+ "\n",
+ "In the example below, the IntRangeWidget is never rendered since display was called on the parent before the parent/child relationship was established."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.MulticontainerWidget()\n",
+ "display(container)\n",
+ "\n",
+ "intrange = widgets.IntRangeWidget(parent=container) # Never gets displayed."
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 3
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Calling display on the child fixes the problem."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.MulticontainerWidget()\n",
+ "display(container)\n",
+ "\n",
+ "intrange = widgets.IntRangeWidget(parent=container)\n",
+ "display(intrange) # This line is needed since the `container` has already been displayed."
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 4
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Changing Child Views"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The view used to display a widget must defined by the time the widget is displayed. If children widgets are to be displayed along with the parent in one call, their `view_name`s can't be set since all of the widgets are sharing the same display call. Instead, their `default_view_name`s must be set (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.MulticontainerWidget()\n",
+ "\n",
+ "floatrange = widgets.FloatRangeWidget(parent=container)\n",
+ "floatrange.default_view_name = \"FloatTextView\" # It can be set as a property.\n",
+ "\n",
+ "string = widgets.StringWidget(default_view_name = \"TextAreaView\") # It can also be set in the constructor.\n",
+ "string.parent = container\n",
+ "\n",
+ "display(container)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 5
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "However, if the children are displayed after the parent, their `view_name` can also be set like normal. Both methods will work. The code below produces the same output as the code above."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.MulticontainerWidget()\n",
+ "display(container)\n",
+ "\n",
+ "floatrange = widgets.FloatRangeWidget()\n",
+ "floatrange.parent=container\n",
+ "display(floatrange, view_name = \"FloatTextView\") # view_name can be set during display.\n",
+ "\n",
+ "string = widgets.StringWidget()\n",
+ "string.parent = container\n",
+ "string.default_view_name = \"TextAreaView\" # Setting default_view_name still works.\n",
+ "display(string)\n"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 6
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "Visibility"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Sometimes it's necessary to hide/show widget views in place, without ruining the order that they have been displayed on the page. Using the `display` method, the views are always added to the end of their respective containers. Instead the `visibility` property of widgets can be used to hide/show widgets that have already been displayed (as seen below)."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "string = widgets.StringWidget(value=\"Hello World!\")\n",
+ "display(string, view_name=\"LabelView\") "
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 7
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "string.visible=False"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 8
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "string.visible=True"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 9
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "In the example below, a form is rendered which conditionally displays widgets depending on the state of other widgets. Try toggling the student checkbox."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "form = widgets.ContainerWidget()\n",
+ "first = widgets.StringWidget(description=\"First Name:\", parent=form)\n",
+ "last = widgets.StringWidget(description=\"Last Name:\", parent=form)\n",
+ "\n",
+ "student = widgets.BoolWidget(description=\"Student:\", value=False, parent=form)\n",
+ "school_info = widgets.ContainerWidget(visible=False, parent=form)\n",
+ "school = widgets.StringWidget(description=\"School:\", parent=school_info)\n",
+ "grade = widgets.IntRangeWidget(description=\"Grade:\", min=0, max=12, default_view_name='IntTextView', parent=school_info)\n",
+ "\n",
+ "pet = widgets.StringWidget(description=\"Pet's Name:\", parent=form)\n",
+ "display(form)\n",
+ "\n",
+ "def on_student_toggle(name, value):\n",
+ " if value:\n",
+ " school_info.visible = True\n",
+ " else:\n",
+ " school_info.visible = False\n",
+ "student.on_trait_change(on_student_toggle, 'value')\n"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 10
+ }
+ ],
+ "metadata": {}
+ }
+ ]
+}
\ No newline at end of file
diff --git a/examples/notebooks/widgets/Tutorials - Styles.ipynb b/examples/notebooks/widgets/Tutorials - Styles.ipynb
new file mode 100644
index 0000000..bdd7168
--- /dev/null
+++ b/examples/notebooks/widgets/Tutorials - Styles.ipynb
@@ -0,0 +1,405 @@
+{
+ "metadata": {
+ "name": ""
+ },
+ "nbformat": 3,
+ "nbformat_minor": 0,
+ "worksheets": [
+ {
+ "cells": [
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "from IPython.html import widgets # Widget definitions\n",
+ "from IPython.display import display # Used to display widgets in the notebook\n",
+ "\n",
+ "# Enable widgets in this notebook\n",
+ "widgets.init_widget_js()"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/bool.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/int_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/int.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/selection.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/string.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/float.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/container.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/multicontainer.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/button.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ },
+ {
+ "javascript": [
+ "$.getScript(\"../static/notebook/js/widgets/float_range.js\");"
+ ],
+ "metadata": {},
+ "output_type": "display_data"
+ }
+ ],
+ "prompt_number": 1
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "CSS"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "When trying to design an attractive widget GUI, styling becomes important. Widget views are DOM (document object model) elements that can be controlled with CSS. There are two helper methods defined on widget that allow the manipulation of the widget's CSS. The first is the `set_css` method, whos doc string is displayed below. This method allows one or more CSS attributes to be set at once. "
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "print(widgets.Widget.set_css.__doc__)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Set one or more CSS properties of the widget (shared among all of the \n",
+ " views). This function has two signatures:\n",
+ " - set_css(css_dict, [selector=''])\n",
+ " - set_css(key, value, [selector=''])\n",
+ "\n",
+ " Parameters\n",
+ " ----------\n",
+ " css_dict : dict\n",
+ " CSS key/value pairs to apply\n",
+ " key: unicode\n",
+ " CSS key\n",
+ " value\n",
+ " CSS value\n",
+ " selector: unicode (optional)\n",
+ " JQuery selector to use to apply the CSS key/value.\n",
+ " \n"
+ ]
+ }
+ ],
+ "prompt_number": 2
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The second is `get_css` which allows CSS attributes that have been set to be read. Note that this method will only read CSS attributes that have been set using the `set_css` method. `get_css`'s doc string is displayed below."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "print(widgets.Widget.get_css.__doc__)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Get a CSS property of the widget. Note, this function does not \n",
+ " actually request the CSS from the front-end; Only properties that have \n",
+ " been set with set_css can be read.\n",
+ "\n",
+ " Parameters\n",
+ " ----------\n",
+ " key: unicode\n",
+ " CSS key\n",
+ " selector: unicode (optional)\n",
+ " JQuery selector used when the CSS key/value was set.\n",
+ " \n"
+ ]
+ }
+ ],
+ "prompt_number": 3
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Below is an example that applies CSS attributes to a container to emphasize text."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.ContainerWidget()\n",
+ "\n",
+ "# set_css used to set a single CSS attribute.\n",
+ "container.set_css('border', '3px solid black') # Border the container\n",
+ "\n",
+ "# set_css used to set multiple CSS attributes.\n",
+ "container.set_css({'padding': '6px', # Add padding to the container\n",
+ " 'background': 'yellow'}) # Fill the container yellow\n",
+ "\n",
+ "label = widgets.StringWidget(default_view_name=\"LabelView\", parent=container)\n",
+ "label.value = \"ALERT: Hello World!\"\n",
+ "\n",
+ "display(container)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 4
+ },
+ {
+ "cell_type": "heading",
+ "level": 1,
+ "metadata": {},
+ "source": [
+ "DOM Classes"
+ ]
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "In some cases it's necessary to apply DOM classes to your widgets. DOM classes allow DOM elements to be indentified by Javascript and CSS. The notebook defines its own set of classes to stylize its elements. The `add_class` widget method allows you to add DOM classes to your widget's definition. The `add_class` method's doc string can be seen below."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "print(widgets.Widget.add_class.__doc__)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Add class[es] to a DOM element\n",
+ "\n",
+ " Parameters\n",
+ " ----------\n",
+ " class_name: unicode\n",
+ " Class name(s) to add to the DOM element(s). Multiple class names \n",
+ " must be space separated.\n",
+ " selector: unicode (optional)\n",
+ " JQuery selector to select the DOM element(s) that the class(es) will \n",
+ " be added to.\n",
+ " \n"
+ ]
+ }
+ ],
+ "prompt_number": 5
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "Since `add_class` if a DOM operation, it will only affect widgets that have been displayed. `add_class` must be called after the widget has been displayed. Extending the example above, the corners of the container can be rounded by adding the `corner-all` notebook class to the container (as seen below). "
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "container = widgets.ContainerWidget()\n",
+ "container.set_css({'border': '3px solid black',\n",
+ " 'padding': '6px',\n",
+ " 'background': 'yellow'}) \n",
+ "\n",
+ "label = widgets.StringWidget(default_view_name=\"LabelView\", parent=container) \n",
+ "label.value = \"ALERT: Hello World!\"\n",
+ "\n",
+ "display(container)\n",
+ "container.add_class('corner-all') # Must be called AFTER display"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 6
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The IPython notebook uses bootstrap for styling. The example above can be simplified by using a bootstrap class (as seen below). Bootstrap documentation can be found at http://getbootstrap.com/\u200e ."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "label = widgets.StringWidget(value = \"ALERT: Hello World!\")\n",
+ "display(label, view_name=\"LabelView\")\n",
+ "\n",
+ "# Apply twitter bootstrap alert class to the label.\n",
+ "label.add_class(\"alert\")"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 7
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The example below shows how bootstrap classes can be used to change button apearance."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "# List of the bootstrap button styles\n",
+ "button_classes = ['Default', 'btn-primary', 'btn-info', 'btn-success', \n",
+ " 'btn-warning', 'btn-danger', 'btn-inverse', 'btn-link']\n",
+ "\n",
+ "# Create each button and apply the style. Also add margin to the buttons so they space\n",
+ "# themselves nicely.\n",
+ "for i in range(8):\n",
+ " button = widgets.ButtonWidget(description=button_classes[i])\n",
+ " button.set_css(\"margin\", \"5px\")\n",
+ " display(button)\n",
+ " if i > 0: # Don't add a class the first button.\n",
+ " button.add_class(button_classes[i])\n",
+ " "
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 8
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "It's also useful to be able to remove DOM classes from widgets. The `remove_class` widget method allows you to remove classes from widgets that have been displayed. Like `add_widget`, it must be called after the widget has been displayed. The doc string of `remove_class` can be seen below."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "print(widgets.Widget.remove_class.__doc__)"
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [
+ {
+ "output_type": "stream",
+ "stream": "stdout",
+ "text": [
+ "Remove class[es] from a DOM element\n",
+ "\n",
+ " Parameters\n",
+ " ----------\n",
+ " class_name: unicode\n",
+ " Class name(s) to remove from the DOM element(s). Multiple class \n",
+ " names must be space separated.\n",
+ " selector: unicode (optional)\n",
+ " JQuery selector to select the DOM element(s) that the class(es) will \n",
+ " be removed from.\n",
+ " \n"
+ ]
+ }
+ ],
+ "prompt_number": 9
+ },
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "The example below animates an alert using different bootstrap styles."
+ ]
+ },
+ {
+ "cell_type": "code",
+ "collapsed": false,
+ "input": [
+ "import time\n",
+ "label = widgets.StringWidget(value = \"ALERT: Hello World!\")\n",
+ "display(label, view_name=\"LabelView\")\n",
+ "\n",
+ "# Apply twitter bootstrap alert class to the label.\n",
+ "label.add_class(\"alert\")\n",
+ "\n",
+ "# Animate through additional bootstrap label styles 3 times\n",
+ "additional_alert_styles = ['alert-error', 'alert-info', 'alert-success']\n",
+ "for i in range(3 * len(additional_alert_styles)):\n",
+ " label.add_class(additional_alert_styles[i % 3])\n",
+ " label.remove_class(additional_alert_styles[(i-1) % 3])\n",
+ " time.sleep(1)\n",
+ " "
+ ],
+ "language": "python",
+ "metadata": {},
+ "outputs": [],
+ "prompt_number": 10
+ }
+ ],
+ "metadata": {}
+ }
+ ]
+}
\ No newline at end of file