From 25beae438f579fc2358f745b5955b01a6312b377 Mon Sep 17 00:00:00 2001 From: "Brian E. Granger" Date: Mon, 14 Apr 2014 16:03:47 -0700 Subject: [PATCH 01/28] Creating overall structure and organization. --- .../Notebook/Custom Keyboard Shortcuts.ipynb | 9 ++++++ examples/Notebook/Index.ipynb | 32 +++++++++++-------- ...ce.ipynb => Notebook User Interface.ipynb} | 0 .../Running the Notebook Server.ipynb | 9 ++++++ ...hJax.ipynb => Typesetting Equations.ipynb} | 0 .../What is the IPython Notebook?.ipynb | 9 ++++++ ...de.ipynb => Working With Code Cells.ipynb} | 0 ...pynb => Working With Markdown Cells.ipynb} | 0 8 files changed, 46 insertions(+), 13 deletions(-) create mode 100644 examples/Notebook/Custom Keyboard Shortcuts.ipynb rename examples/Notebook/{User Interface.ipynb => Notebook User Interface.ipynb} (100%) create mode 100644 examples/Notebook/Running the Notebook Server.ipynb rename examples/Notebook/{Typesetting Math Using MathJax.ipynb => Typesetting Equations.ipynb} (100%) create mode 100644 examples/Notebook/What is the IPython Notebook?.ipynb rename examples/Notebook/{Running Code.ipynb => Working With Code Cells.ipynb} (100%) rename examples/Notebook/{Markdown Cells.ipynb => Working With Markdown Cells.ipynb} (100%) diff --git a/examples/Notebook/Custom Keyboard Shortcuts.ipynb b/examples/Notebook/Custom Keyboard Shortcuts.ipynb new file mode 100644 index 000000000..519dba713 --- /dev/null +++ b/examples/Notebook/Custom Keyboard Shortcuts.ipynb @@ -0,0 +1,9 @@ +{ + "metadata": { + "name": "", + "signature": "sha256:0abf067a20ebda26a671db997ac954770350d292dff7b7d6a4ace8808f70aca1" + }, + "nbformat": 3, + "nbformat_minor": 0, + "worksheets": [] +} \ No newline at end of file diff --git a/examples/Notebook/Index.ipynb b/examples/Notebook/Index.ipynb index 1454ce0a0..07f160abc 100644 --- a/examples/Notebook/Index.ipynb +++ b/examples/Notebook/Index.ipynb @@ -1,7 +1,7 @@ { "metadata": { "name": "", - "signature": "sha256:c9fdf35bcd07c381b988af18346fceeb08107ce216817e4bd398641e9c93ecc1" + "signature": "sha256:c81ae6aa03abb91a6a64807c6cd7dae22f140f16780d54f0eb4beb3b1f8606aa" }, "nbformat": 3, "nbformat_minor": 0, @@ -49,14 +49,14 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "* [User Interface](User Interface.ipynb)\n", - "* [Running Code](Running Code.ipynb)\n", - "* [Basic Output](Basic Output.ipynb)\n", - "* [Plotting with Matplotlib](Plotting with Matplotlib.ipynb)\n", - "* [Markdown Cells](Markdown Cells.ipynb)\n", - "* [Typesetting Math Using MathJax](Typesetting Math Using MathJax.ipynb)\n", - "* [Display System](Display System.ipynb)\n", - "* [Custom Display Logic](Custom Display Logic.ipynb)" + "* [What is the IPython Notebook?](What is the IPython Notebook%3F.ipynb)\n", + "* [Running the Notebook Server](Running the Notebook Server.ipynb)\n", + "* [Notebook Dashboard](Notebook Dashboard.ipynb)\n", + "* [Notebook User Interface](Notebook User Interface.ipynb)\n", + "* [Notebook Cell Types](Notebook Cell Types.ipynb)\n", + "* [Working With Code Cells](Working With Code Cells.ipynb)\n", + "* [Working With Markdown Cells](Working With Markdown Cells.ipynb)\n", + "* [Notebook Security](Notebook Security.ipynb)" ] }, { @@ -71,13 +71,19 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "* [Trapezoid Rule](Trapezoid Rule.ipynb)\n", - "* [SymPy](SymPy.ipynb)\n", - "* [Raw Input](Raw Input.ipynb)\n", + "* [Custom Keyboard Shortcuts](Custom Keyboard Shortcuts.ipynb)\n", "* [Importing Notebooks](Importing Notebooks.ipynb)\n", "* [Connecting with the Qt Console](Connecting with the Qt Console.ipynb)\n", - "* [Animations Using clear_output](Animations Using clear_output.ipynb)" + "* [Typesetting Equations](Typesetting Equations.ipynb)" ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [], + "language": "python", + "metadata": {}, + "outputs": [] } ], "metadata": {} diff --git a/examples/Notebook/User Interface.ipynb b/examples/Notebook/Notebook User Interface.ipynb similarity index 100% rename from examples/Notebook/User Interface.ipynb rename to examples/Notebook/Notebook User Interface.ipynb diff --git a/examples/Notebook/Running the Notebook Server.ipynb b/examples/Notebook/Running the Notebook Server.ipynb new file mode 100644 index 000000000..519dba713 --- /dev/null +++ b/examples/Notebook/Running the Notebook Server.ipynb @@ -0,0 +1,9 @@ +{ + "metadata": { + "name": "", + "signature": "sha256:0abf067a20ebda26a671db997ac954770350d292dff7b7d6a4ace8808f70aca1" + }, + "nbformat": 3, + "nbformat_minor": 0, + "worksheets": [] +} \ No newline at end of file diff --git a/examples/Notebook/Typesetting Math Using MathJax.ipynb b/examples/Notebook/Typesetting Equations.ipynb similarity index 100% rename from examples/Notebook/Typesetting Math Using MathJax.ipynb rename to examples/Notebook/Typesetting Equations.ipynb diff --git a/examples/Notebook/What is the IPython Notebook?.ipynb b/examples/Notebook/What is the IPython Notebook?.ipynb new file mode 100644 index 000000000..519dba713 --- /dev/null +++ b/examples/Notebook/What is the IPython Notebook?.ipynb @@ -0,0 +1,9 @@ +{ + "metadata": { + "name": "", + "signature": "sha256:0abf067a20ebda26a671db997ac954770350d292dff7b7d6a4ace8808f70aca1" + }, + "nbformat": 3, + "nbformat_minor": 0, + "worksheets": [] +} \ No newline at end of file diff --git a/examples/Notebook/Running Code.ipynb b/examples/Notebook/Working With Code Cells.ipynb similarity index 100% rename from examples/Notebook/Running Code.ipynb rename to examples/Notebook/Working With Code Cells.ipynb diff --git a/examples/Notebook/Markdown Cells.ipynb b/examples/Notebook/Working With Markdown Cells.ipynb similarity index 100% rename from examples/Notebook/Markdown Cells.ipynb rename to examples/Notebook/Working With Markdown Cells.ipynb From 2321abc143bf0d0bd08b2bb3e0b58f9af7e879a9 Mon Sep 17 00:00:00 2001 From: "Brian E. Granger" Date: Tue, 15 Apr 2014 10:08:05 -0700 Subject: [PATCH 02/28] Work on notebook docs. --- examples/Notebook/Index.ipynb | 4 +- .../What is the IPython Notebook?.ipynb | 124 +++++++++++++++++- 2 files changed, 124 insertions(+), 4 deletions(-) diff --git a/examples/Notebook/Index.ipynb b/examples/Notebook/Index.ipynb index 07f160abc..61c9c8402 100644 --- a/examples/Notebook/Index.ipynb +++ b/examples/Notebook/Index.ipynb @@ -1,7 +1,7 @@ { "metadata": { "name": "", - "signature": "sha256:c81ae6aa03abb91a6a64807c6cd7dae22f140f16780d54f0eb4beb3b1f8606aa" + "signature": "sha256:f3479d16d8a60d0aa82e5fbffd52552e5649074a7a660730ddca9d5b286692b2" }, "nbformat": 3, "nbformat_minor": 0, @@ -34,7 +34,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "The IPython Notebook is a web-based interactive computing system that enables users to author documents that include live code, narrative text, LaTeX equations, HTML, images and video. These documents are contain a full record of a computation and its results and can be shared on email, [Dropbox](http://dropbox.com), version control systems (like git/[GitHub](http://github.com)) or [nbviewer.ipython.org](http://nbviewer.ipython.org)." + "The IPython Notebook is a web-based interactive computing system that enables users to author documents that include live code, narrative text, LaTeX equations, HTML, images and video. These documents contain a full record of a computation and its results and can be shared on email, [Dropbox](http://dropbox.com), version control systems (like git/[GitHub](http://github.com)) or [nbviewer.ipython.org](http://nbviewer.ipython.org)." ] }, { diff --git a/examples/Notebook/What is the IPython Notebook?.ipynb b/examples/Notebook/What is the IPython Notebook?.ipynb index 519dba713..7275086a9 100644 --- a/examples/Notebook/What is the IPython Notebook?.ipynb +++ b/examples/Notebook/What is the IPython Notebook?.ipynb @@ -1,9 +1,129 @@ { "metadata": { "name": "", - "signature": "sha256:0abf067a20ebda26a671db997ac954770350d292dff7b7d6a4ace8808f70aca1" + "signature": "sha256:3676a017d63581291cf2d61ea56d09cb3e5480a4f222184d8216e3e829eb4871" }, "nbformat": 3, "nbformat_minor": 0, - "worksheets": [] + "worksheets": [ + { + "cells": [ + { + "cell_type": "heading", + "level": 1, + "metadata": {}, + "source": [ + "What is the IPython Notebook?" + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Introduction" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The IPython Notebook is an interactive computing environment that enables users to author notebook documents that include live code, interactive widgets, plots, narrative text, equations, images and video. These documents provide a complete and self-contained record of a computation that can be converted to various formats and shared with others using email, [Dropbox](http://dropbox.com), version control systems (like git/[GitHub](http://github.com)) or [nbviewer.ipython.org](http://nbviewer.ipython.org).\n", + "\n", + "Through IPython's kernel and messaging architecture, the Notebook allows code to be run in a range of different programming languages, including Python, R, Julia, Ruby and many others.\n", + "\n", + "The IPython Notebook combines three components:\n", + "\n", + "* **The notebook web application**: An interactive web application for writing and running code interactively and authoring notebook documents.\n", + "* **Kernels**: Separate processes started by the notebook web application that runs users' code in a given language and returns output back to the notebook web application. The kernel also handles things like computations for interactive widgets, tab completion and introspection. \n", + "* **Notebook documents**: Self-contained documents that contain a representation of all content visible in the notebook web application, including inputs and outputs of the computations, narrative\n", + "text, equations, images, and rich media representations of objects. Each notebook document has its own kernel." + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Notebook web application" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The notebook web application enables users to:\n", + "\n", + "* Edit code in the browser, with automatic syntax highlighting, indentation, and tab completion/introspection.\n", + "* Run code from the browser, with the results of computations attached to the code which generated them.\n", + "* See the results of computations with rich media representations, such as HTML, LaTeX, PNG, SVG, PDF, etc.\n", + "* Create and use interactive JavaScript wigets, which bind interactive user interface controls and visualizations to reactive kernel side computations.\n", + "* Author narrative text using the [Markdown](https://daringfireball.net/projects/markdown/) markup language.\n", + "* Build hierarchical documents that are organized into sections with different levels of headings.\n", + "* Include mathematical equations using LaTeX syntax in Markdown, which are rendered in-browser by [MathJax](http://www.mathjax.org/).\n", + "* Start parallel computing clusters that work with IPython's interactive parallel computing libraries `IPython.parallel`." + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Kernels" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "For each notebook document that a user opens, the web application starts a kernel that runs the code for that notebook. Each kernel is capable of running code in a single programming language and there are kernels available in the following languages:\n", + "\n", + "* Python(https://github.com/ipython/ipython)\n", + "* Julia (https://github.com/JuliaLang/IJulia.jl)\n", + "* R (https://github.com/takluyver/IRkernel)\n", + "* Ruby (https://github.com/minrk/iruby)\n", + "* Haskell (https://github.com/gibiansky/IHaskell)\n", + "* Scala (https://github.com/Bridgewater/scala-notebook)\n", + "* node.js (https://gist.github.com/Carreau/4279371)\n", + "* Go (https://github.com/takluyver/igo)\n", + "\n", + "The default kernel runs Python code. When it is released in the Summer/Fall of 2014, IPython 3.0 will provide a simple way for users to pick which of these kernels is used for a given notebook. \n", + "\n", + "Each of these kernels communicate with the notebook web application and web browser using a JSON over ZeroMQ/WebSockets message protocol that is described [here](http://ipython.org/ipython-doc/dev/development/messaging.html). Most users don't need to know about these details, but it helps to understand that \"kernels run code.\"" + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Notebook documents" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Notebook documents contain the inputs and outputs of an interactive session as well as narrative text that accompanies the code but is not meant for execution. Rich output generated by running code, including HTML, images, video, and plots, is embeddeed in the notebook, which makes it a complete and self-contained record of a computation. \n", + "\n", + "When you run the notebook web application on your computer, notebook documents are just files on your local filesystem with a `.ipynb` extension. This allows you to use familiar workflows for organizing your notebooks into folders and sharing them with others using email, Dropbox and version control systems.\n", + "\n", + "Notebooks consist of a linear sequence of cells. There are four basic cell types:\n", + "\n", + "* **Code cells:** Input and output of live code that is run in the kernel\n", + "* **Markdown cells:** Narrative text with embedded LaTeX equations\n", + "* **Heading cells:** 6 levels of hierarchical organization\n", + "* **Raw cells:** Unformatted text that is included, without modification, when notebooks are converted to different formats using nbconvert\n", + "\n", + "Internally, notebook documents are [JSON](http://en.wikipedia.org/wiki/JSO) data with binary values [base64](http://en.wikipedia.org/wiki/Base64) encoded. This allows them to be read and manipulated programmatically by any programming language. Because JSON is a text format, notebook documents are version control friendly.\n", + "\n", + "Notebooks can be exported to different static formats including HTML, reStructeredText, LaTeX, PDF, and slide shows ([reveal.js](http://lab.hakim.se/reveal-js/#/)) using IPython's `nbconvert` utility.\n", + "\n", + "Furthermore, any notebook document available from a public URL on or GitHub can be shared via http://nbviewer.ipython.org. This service loads the notebook document from the URL and renders it as a static web page. The resulting web page may thus be shared with others without their needing to install IPython." + ] + } + ], + "metadata": {} + } + ] } \ No newline at end of file From 99acef68a6df67de1653c04183d5b9484dbb1ab0 Mon Sep 17 00:00:00 2001 From: "Brian E. Granger" Date: Tue, 29 Apr 2014 12:25:26 -0700 Subject: [PATCH 03/28] Working on notebook docs. --- .../Running the Notebook Server.ipynb | 243 +++++++++++++++++- 1 file changed, 241 insertions(+), 2 deletions(-) diff --git a/examples/Notebook/Running the Notebook Server.ipynb b/examples/Notebook/Running the Notebook Server.ipynb index 519dba713..26daeab87 100644 --- a/examples/Notebook/Running the Notebook Server.ipynb +++ b/examples/Notebook/Running the Notebook Server.ipynb @@ -1,9 +1,248 @@ { "metadata": { "name": "", - "signature": "sha256:0abf067a20ebda26a671db997ac954770350d292dff7b7d6a4ace8808f70aca1" + "signature": "sha256:84fa8b285a29d021b31d7211c6452729fb042a5c74cee6bf31e74e06fdd26b97" }, "nbformat": 3, "nbformat_minor": 0, - "worksheets": [] + "worksheets": [ + { + "cells": [ + { + "cell_type": "heading", + "level": 1, + "metadata": {}, + "source": [ + "Running the Notebook Server" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The IPython notebook server is a custom web server that runs the notebook web application. Most of the time, users run the notebook server on their local computer using IPython's command line interface." + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Starting the notebook server using the command line" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You can start the notebook server from the command line (Terminal on Mac/Linux, CMD prompt on Windows) by running the following command: \n", + "\n", + " ipython notebook\n", + "\n", + "This will print some information about the notebook server in your console, including the URL of the web application (by default, `http://127.0.0.1:8888`). It will then open your default web browser to this URL.\n", + "\n", + "When the notebook opens, you will see the **notebook dashboard**, which will show a list of the notebooks and subdirectories in the directory where the notebook server was started. As of IPython 2.0, the dashboard allows you to navigate to different subdirectories. Because of this, it is no longer necessary to start a separate notebook server for each subdirectory. Most of the time, you will want to start a notebook server in the highest directory in your filesystem where notebooks can be found. Often this will be your home directory.\n", + "\n", + "You can start more than one notebook server at the same time. By default, the first notebook server starts on port 8888 and later notebook servers search for open ports near that one.\n", + "\n", + "You can also specify the port manually:\n", + "\n", + " ipython notebook --port 9999\n", + "\n", + "Or start notebook server without opening a web browser.\n", + "\n", + " ipython notebook --no-browser\n", + "\n", + "The notebook server has a number of other command line arguments that can be displayed with the `--help` flag: \n", + "\n", + " ipython notebook --help\n", + "\n", + "
\n", + "It used to be possible to specify kernel options, such as --pylab inline from the command line. This is deprecated in IPython 2.0 and will be removed in IPython 3.0. To enable matplotlib based plotting for the Python kernel use the %matplotlib magic command.\n", + "
\n", + "\n" + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Configuring the IPython Notebook" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The notebook web server can also be configured using IPython profiles and configuration files. The Notebook web server configuration options are set in a file named `ipython_notebook_config.py` in your IPython *profile directory*. The profile directory is a subfolder of your IPython directory, which itself is usually `.ipython` in your home directory.\n", + "\n", + "You can display the location of your default profile directory by running the command:\n", + "\n", + " ipython locate\n", + "\n", + "The default version of `ipython_notebook_config.py` lists all of the options available along with documentation for each. Changes made to that file will affect all notebook servers run under that profile. Command line options always override those set in configuration files.\n", + "\n", + "More details about IPython configuration files and profiles can be found [here](http://ipython.org/ipython-doc/dev/config/intro.html)." + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Securing the notebook server" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The IPython Notebook allows arbitrary code execution on the computer running it. Thus, the notebook web server should never be run on the open internet without first securing it. By default, the notebook server only listens on local network interface (`127.0.0.1`) There are two steps required to secure the notebook server:\n", + "\n", + "1. Setting a password\n", + "2. Encrypt network traffic using SSL\n", + "\n", + "\n", + "\n", + "You can protect your notebook server with a simple single password by setting the `NotebookApp.password` configurable. You can prepare a hashed password using the function `IPython.lib.security.passwd`:\n", + "\n", + "```python\n", + "In [1]: from IPython.lib import passwd\n", + "In [2]: passwd()\n", + "Enter password: \n", + "Verify password: \n", + "Out[2]: 'sha1:67c9e60bb8b6:9ffede0825894254b2e042ea597d771089e11aed'\n", + "```\n", + "\n", + "
\n", + "`IPython.lib.security.passwd` can also take the password as a string argument. **Do not** pass it as an argument inside an IPython session, as it will be saved in your input history.\n", + "
\n", + "\n", + "You can then add this to your `ipython_notebook_config.py`:\n", + "\n", + "```python\n", + "# Password to use for web authentication\n", + "c = get_config()\n", + "c.NotebookApp.password = \n", + "u'sha1:67c9e60bb8b6:9ffede0825894254b2e042ea597d771089e11aed'\n", + "```\n", + "When using a password, it is a good idea to also use SSL, so that your \n", + "password is not sent unencrypted by your browser. You can start the notebook \n", + "to communicate via a secure protocol mode using a self-signed certificate with \n", + "the command::\n", + "\n", + " $ ipython notebook --certfile=mycert.pem\n", + "\n", + ".. note::\n", + "\n", + " A self-signed certificate can be generated with ``openssl``. For example, \n", + " the following command will create a certificate valid for 365 days with \n", + " both the key and certificate data written to the same file::\n", + "\n", + " $ openssl req -x509 -nodes -days 365 -newkey rsa:1024 -keyout mycert.pem -out mycert.pem\n", + "\n", + "Your browser will warn you of a dangerous certificate because it is\n", + "self-signed. If you want to have a fully compliant certificate that will not\n", + "raise warnings, it is possible (but rather involved) to obtain one,\n", + "as explained in detail in `this tutorial`__.\n", + "\n", + ".. __: http://arstechnica.com/security/news/2009/12/how-to-get-set-with-a-secure-sertificate-for-free.ars\n", + "\t\n", + "Keep in mind that when you enable SSL support, you will need to access the\n", + "notebook server over ``https://``, not over plain ``http://``. The startup\n", + "message from the server prints this, but it is easy to overlook and think the\n", + "server is for some reason non-responsive." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Running a public notebook server\n", + "--------------------------------\n", + "\n", + "If you want to access your notebook server remotely via a web browser,\n", + "you can do the following. \n", + "\n", + "Start by creating a certificate file and a hashed password, as explained \n", + "above. Then create a custom profile for the notebook, with the following \n", + "command line, type::\n", + "\n", + " $ ipython profile create nbserver\n", + "\n", + "In the profile directory just created, edit the file \n", + "``ipython_notebook_config.py``. By default, the file has all fields \n", + "commented; the minimum set you need to uncomment and edit is the following::\n", + "\n", + " c = get_config()\n", + "\n", + " # Kernel config\n", + " c.IPKernelApp.pylab = 'inline' # if you want plotting support always\n", + "\n", + " # Notebook config\n", + " c.NotebookApp.certfile = u'/absolute/path/to/your/certificate/mycert.pem'\n", + " c.NotebookApp.ip = '*'\n", + " c.NotebookApp.open_browser = False\n", + " c.NotebookApp.password = u'sha1:bcd259ccf...[your hashed password here]'\n", + " # It is a good idea to put it on a known, fixed port\n", + " c.NotebookApp.port = 9999\n", + "\n", + "You can then start the notebook and access it later by pointing your browser \n", + "to ``https://your.host.com:9999`` with ``ipython notebook \n", + "--profile=nbserver``.\n", + "\n", + "Running with a different URL prefix\n", + "-----------------------------------\n", + "\n", + "The notebook dashboard (the landing page with an overview\n", + "of the notebooks in your working directory) typically lives at the URL\n", + "``http://localhost:8888/``. If you prefer that it lives, together with the \n", + "rest of the notebook, under a sub-directory,\n", + "e.g. ``http://localhost:8888/ipython/``, you can do so with\n", + "configuration options like the following (see above for instructions about\n", + "modifying ``ipython_notebook_config.py``)::\n", + "\n", + " c.NotebookApp.base_url = '/ipython/'\n", + " c.NotebookApp.webapp_settings = {'static_url_prefix':'/ipython/static/'}\n", + "\n", + "Using a different notebook store\n", + "--------------------------------\n", + "\n", + "By default, the notebook server stores the notebook documents that it saves as \n", + "files in the working directory of the notebook server, also known as the\n", + "``notebook_dir``. This logic is implemented in the \n", + ":class:`FileNotebookManager` class. However, the server can be configured to \n", + "use a different notebook manager class, which can \n", + "store the notebooks in a different format. \n", + "\n", + "The bookstore_ package currently allows users to store notebooks on Rackspace\n", + "CloudFiles or OpenStack Swift based object stores.\n", + "\n", + "Writing a notebook manager is as simple as extending the base class\n", + ":class:`NotebookManager`. The simple_notebook_manager_ provides a great example\n", + "of an in memory notebook manager, created solely for the purpose of\n", + "illustrating the notebook manager API.\n", + "\n", + ".. _bookstore: https://github.com/rgbkrk/bookstore\n", + "\n", + ".. _simple_notebook_manager: https://github.com/khinsen/simple_notebook_manager\n", + "\n", + "Known issues\n", + "------------\n", + "\n", + "When behind a proxy, especially if your system or browser is set to autodetect\n", + "the proxy, the notebook web application might fail to connect to the server's\n", + "websockets, and present you with a warning at startup. In this case, you need\n", + "to configure your system not to use the proxy for the server's address.\n", + "\n", + "For example, in Firefox, go to the Preferences panel, Advanced section,\n", + "Network tab, click 'Settings...', and add the address of the notebook server\n", + "to the 'No proxy for' field." + ] + } + ], + "metadata": {} + } + ] } \ No newline at end of file From 7d8fe7e37d2bc908920c2321ee78cc24e0dbb9a8 Mon Sep 17 00:00:00 2001 From: "Brian E. Granger" Date: Tue, 29 Apr 2014 14:55:44 -0700 Subject: [PATCH 04/28] Moving over new interact tutorial. --- .../Interactive Widgets/Using Interact.ipynb | 716 ++++++++++++++++-- 1 file changed, 658 insertions(+), 58 deletions(-) diff --git a/examples/Interactive Widgets/Using Interact.ipynb b/examples/Interactive Widgets/Using Interact.ipynb index 7190534b5..e6312f8c8 100644 --- a/examples/Interactive Widgets/Using Interact.ipynb +++ b/examples/Interactive Widgets/Using Interact.ipynb @@ -1,7 +1,7 @@ { "metadata": { "name": "", - "signature": "sha256:3f30c6e839ac39f890da34a2af6bf50bf0d99ea32f7aadc043f3e31f619e4bc9" + "signature": "sha256:6d8c7c51322c4911e478068e8fa8e897bd72c614096f5df110ed86d01d66001c" }, "nbformat": 3, "nbformat_minor": 0, @@ -13,14 +13,14 @@ "level": 1, "metadata": {}, "source": [ - "Interact" + "Using Interact" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "The `interact` function provides a high-level interface for creating user interface controls to use in exploring code and data interactively." + "The `interact` function (`IPython.html.widgets.interact`) automatically creates user interface (UI) controls for exploring code and data interactively. It is the easiest way to get started using IPython's widgets." ] }, { @@ -28,66 +28,70 @@ "collapsed": false, "input": [ "from IPython.html.widgets import interact, interactive, fixed\n", - "from IPython.html import widgets\n", - "from IPython.display import clear_output, display, HTML" + "from IPython.html import widgets" ], "language": "python", "metadata": {}, "outputs": [], "prompt_number": 1 }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "
\n", + "As of IPython 2.0, the widgets in this notebook won't show up on http://nbviewer.ipython.org. To view the widgets and interact with them, you will need to download this notebook and run it with an IPython Notebook server.\n", + "
" + ] + }, { "cell_type": "heading", "level": 2, "metadata": {}, "source": [ - "Basic interact" + "Basic `interact`" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "Here is a simple function that displays its arguments as an HTML table:" + "At the most basic level, `interact` autogenerates UI controls for function arguments, and then calls the function with those arguments when you manipulate the controls interactively. To use `interact`, you need to define a function that you want to explore. Here is a function that prints its only argument `x`." ] }, { "cell_type": "code", "collapsed": false, "input": [ - "def show_args(**kwargs):\n", - " s = '

Arguments:

\\n'\n", - " for k,v in kwargs.items():\n", - " s += '\\n'.format(k,v)\n", - " s += '
{0}{1}
'\n", - " display(HTML(s))" + "def f(x):\n", + " print x" ], "language": "python", "metadata": {}, "outputs": [], "prompt_number": 2 }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "When you pass this function as the first argument to `interact` along with an integer keyword argument (`x=10`), a slider is generated and bound to the function." + ] + }, { "cell_type": "code", "collapsed": false, "input": [ - "show_args(a=10, b='Hi There', c=True)" + "interact(f, x=10);" ], "language": "python", "metadata": {}, "outputs": [ { - "html": [ - "

Arguments:

\n", - "\n", - "\n", - "\n", - "
a10
cTrue
bHi There
" - ], - "metadata": {}, - "output_type": "display_data", + "output_type": "stream", + "stream": "stdout", "text": [ - "" + "9\n" ] } ], @@ -97,71 +101,667 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "Let's use this function to explore how `interact` works." + "When you move the slider, the function is called and the current value of `x` is printed.\n", + "\n", + "If you pass `True` or `False`, `interact` will generate a checkbox:" ] }, { "cell_type": "code", "collapsed": false, "input": [ - "i = interact(show_args,\n", - " Temp=(0,10),\n", - " Current=(0.,10.,0.01),\n", - " z=True,\n", - " Text=u'Type here!',\n", - " #Algorithm=['This','That','Other'],\n", - " a=widgets.FloatSliderWidget(min=-10.0, max=10.0, step=0.1, value=5.0, description=\"Float (a)\")\n", - " )" + "interact(f, x=True);" ], "language": "python", "metadata": {}, "outputs": [ { - "html": [ - "

Arguments:

\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
Current4.99
TextType here!
zTrue
Temp5
Float (a)5.0
" - ], - "metadata": {}, - "output_type": "display_data", + "output_type": "stream", + "stream": "stdout", "text": [ - "" + "True\n" ] } ], "prompt_number": 4 }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If you pass a string, `interact` will generate a text area." + ] + }, { "cell_type": "code", "collapsed": false, "input": [ - "i.widget" + "interact(f, x='Hi there!');" ], "language": "python", "metadata": {}, "outputs": [ { - "html": [ - "

Arguments:

\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
Current4.99
TextType here!
zTrue
Temp5
Float (a)5.0
" - ], - "metadata": {}, - "output_type": "display_data", + "output_type": "stream", + "stream": "stdout", "text": [ - "" + "Hi there!\n" ] } ], "prompt_number": 5 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "`interact` can also be used as a decorator. This allows you to define a function and interact with it in a single shot. As this example shows, `interact` also works with functions that have multiple arguments." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "@interact(x=True, y=1.0)\n", + "def g(x, y):\n", + " print x, y" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "True 1.0\n" + ] + } + ], + "prompt_number": 6 + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Fixing arguments using `fixed`" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "There are times when you may want to explore a function using `interact`, but fix one or more of its arguments to specific values. This can be accomplished by wrapping values with the `fixed` function." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "def h(p, q):\n", + " print p, q" + ], + "language": "python", + "metadata": {}, + "outputs": [], + "prompt_number": 7 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "When we call `interact`, we pass `fixed(20)` for q to hold it fixed at a value of `20`." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(h, p=5, q=fixed(20));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "5 20\n" + ] + } + ], + "prompt_number": 8 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Notice that a slider is only produced for `p` as the value of `q` is fixed." + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Widget abbreviations" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "When you pass an integer valued keyword argument (`x=10`) to `interact`, it generates an integer valued slider control with a range of $[-10,+3\\times10]$. In this case `10` is an *abbreviation* for an actual slider widget:\n", + "\n", + "```python\n", + "IntSliderWidget(min=-10,max=30,step=1,value=10)\n", + "```\n", + "\n", + "In fact, we can get the same result if we pass this `IntSliderWidget` as the keyword argument for `x`:" + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x=widgets.IntSliderWidget(min=-10,max=30,step=1,value=10));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "10\n" + ] + } + ], + "prompt_number": 9 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "This examples clarifies how `interact` proceses its keyword arguments:\n", + "\n", + "1. If the keyword argument is `Widget` instance with a `value` attribute, that widget is used. Any widget with a `value` attribute can be used, even custom ones.\n", + "2. Otherwise, the value is treated as a *widget abbreviation* that is converted to a widget before it is used.\n", + "\n", + "The following table gives an overview of different widget abbreviations:\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
Keyword argumentWidget
`True` or `False`CheckboxWiget
`'Hi there'`TextareaWidget
`value` or `(min,max)` or `(min,max,step)` if integers are passedIntSliderWidget
`value` or `(min,max)` or `(min,max,step)` if floats are passedFloatSliderWidget
`('orange','apple')` or `{'one':1,'two':2}`DropdownWidget
" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You have seen how the checkbox and textarea widgets work above. Here, more details about the different abbreviations for sliders and dropdowns are given.\n", + "\n", + "If a 2-tuple of integers is passed `(min,max)` a integer valued slider is produced with those minimum and maximum (inclusive) values. In this case, the default step size of `1` is used." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x=(0,4));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "2\n" + ] + } + ], + "prompt_number": 10 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If a 3-tuple of integers is passed `(min,max,step)` the step size can also be set." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x=(0,8,2));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "4\n" + ] + } + ], + "prompt_number": 11 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "A float valued slider is produced if the elements of the tuples are floats. Here the minimum is `0.0`, the maximum is `10.0` and step size is `0.1` (the default)." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x=(0.0,10.0));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "5.0\n" + ] + } + ], + "prompt_number": 12 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The step size can be changed by passing a 3rd element in the tuple." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x=(0.0,10.0,0.01));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "4.99\n" + ] + } + ], + "prompt_number": 13 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "For both integer and float valued sliders, you can pick the initial value of the widget by passing a default keyword argument to the underlying Python function. Here we set the initial value of a float slider to `5.5`." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "@interact(x=(0.0,20.0,0.5))\n", + "def h(x=5.5):\n", + " print x" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "5.5\n" + ] + } + ], + "prompt_number": 14 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Dropdown menus can be produced by passing a tuple of strings. In this case, the strings are both used as the names in the dropdown menu UI and passed to the underlying Python function." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x=('apples','oranges'));" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "apples\n" + ] + } + ], + "prompt_number": 15 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If you want a dropdown menu that passes non-string values to the Python function, you can pass a dictionary. The keys in the dictionary are used for the names in the dropdown menu UI and the values are the arguments that are passed to the underlying Python function." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f, x={'one': 10, 'two': 20});" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "20\n" + ] + } + ], + "prompt_number": 16 + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Using function annotations with `interact`" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If you are using Python 3, you can also specify widget abbreviations using [function annotations](https://docs.python.org/3/tutorial/controlflow.html#function-annotations). This is a convenient approach allows the widget abbreviations to be defined with a function.\n", + "\n", + "Define a function with an checkbox widget abbreviation for the argument `x`." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "def f(x:True):\n", + " print x" + ], + "language": "python", + "metadata": {}, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Then, because the widget abbreviation has already been defined, you can call `interact` with a single argument." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f);" + ], + "language": "python", + "metadata": {}, + "outputs": [] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If you are running Python 2, function annotations can be defined using the `@annotate` function." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "from IPython.utils.py3compat import annotate" + ], + "language": "python", + "metadata": {}, + "outputs": [], + "prompt_number": 50 + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "@annotate(x=True)\n", + "def f(x):\n", + " print x" + ], + "language": "python", + "metadata": {}, + "outputs": [], + "prompt_number": 51 + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "interact(f);" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "True\n" + ] + } + ], + "prompt_number": 52 + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "`interactive`" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In addition to `interact` IPython provides another function, `interactive`, that is useful when you want to reuse the widget that are produced or access the data that is bound to the UI controls." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Here is a function that returns the sum of its two arguments." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "def f(a, b):\n", + " return a+b" + ], + "language": "python", + "metadata": {}, + "outputs": [], + "prompt_number": 18 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Unlike `interact`, `interactive` returns a `Widget` instance rather than immediately displaying the widget." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "w = interactive(f, a=10, b=20)" + ], + "language": "python", + "metadata": {}, + "outputs": [], + "prompt_number": 19 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The widget is a `ContainerWidget`, which is a container for other widgets." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "type(w)" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "metadata": {}, + "output_type": "pyout", + "prompt_number": 20, + "text": [ + "IPython.html.widgets.widget_container.ContainerWidget" + ] + } + ], + "prompt_number": 20 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The children of the `ContainerWidget` are two integer valued sliders produced by the widget abbreviations above." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "w.children" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "metadata": {}, + "output_type": "pyout", + "prompt_number": 21, + "text": [ + "(,\n", + " )" + ] + } + ], + "prompt_number": 21 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "To actually display the widgets, you can use IPython's `display` function." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "from IPython.display import display\n", + "display(w)" + ], + "language": "python", + "metadata": {}, + "outputs": [], + "prompt_number": 22 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "At this point, the UI controls work just like they would if `interact` had been used. You can manipulate them interactively and the function will be called. However, the widget instance returned by `interactive` also give you access to the current keyword arguments and return value of the underlying Python function.\n", + "\n", + "Here are the current keyword arguments. If you rerun this cell after manipulating the sliders, the values will have changed." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "w.kwargs" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "metadata": {}, + "output_type": "pyout", + "prompt_number": 23, + "text": [ + "{u'a': 10, u'b': 20}" + ] + } + ], + "prompt_number": 23 + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Here is the current return value of the function." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "w.result" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "metadata": {}, + "output_type": "pyout", + "prompt_number": 24, + "text": [ + "30" + ] + } + ], + "prompt_number": 24 } ], "metadata": {} From a5805f963142f20869953086a62a673289bc0138 Mon Sep 17 00:00:00 2001 From: "Brian E. Granger" Date: Tue, 29 Apr 2014 15:42:26 -0700 Subject: [PATCH 05/28] Pulling content from ipython-in-depth. --- .../JavaScript Notebook Extensions.ipynb | 770 ++++++++++++++++++ 1 file changed, 770 insertions(+) create mode 100644 examples/Notebook/JavaScript Notebook Extensions.ipynb diff --git a/examples/Notebook/JavaScript Notebook Extensions.ipynb b/examples/Notebook/JavaScript Notebook Extensions.ipynb new file mode 100644 index 000000000..6655bd4a0 --- /dev/null +++ b/examples/Notebook/JavaScript Notebook Extensions.ipynb @@ -0,0 +1,770 @@ +{ + "metadata": { + "name": "" + }, + "nbformat": 3, + "nbformat_minor": 0, + "worksheets": [ + { + "cells": [ + { + "cell_type": "heading", + "level": 1, + "metadata": {}, + "source": [ + "Embrasing web standards" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "One of the main reason that allowed us to developp the current notebook web application \n", + "was to embrase the web technology. \n", + "\n", + "By beeing a pure web application using HTML, Javascript and CSS, the Notebook can get \n", + "all the web technology improvement for free. Thus, as browsers support for different \n", + "media extend, The notebook web app should be able to be compatible without modification. \n", + "\n", + "This is also true with performance of the User Interface as the speed of javascript VM increase. " + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The other advantage of using only web technology is that the code of the interface is fully accessible to the end user, and modifiable live.\n", + "Even if this task is not always easy, we strive to keep our code as accessible and reusable as possible.\n", + "This should allow with minimum effort to develop small extensions that customize the behavior of the web interface. " + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Tempering with Notebook app" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The first tool that is availlable to you and that you shoudl be aware of are browser \"developpers tool\". The exact naming can change across browser, and might require the installation of extensions. But basically they can allow you to inspect/modify the DOM, and interact with the javascript code that run the frontend.\n", + "\n", + " - In Chrome and safari Developper tools are in the menu [Put mmenu name in english here] \n", + " - In firefox you might need to install [Firebug](http://getfirebug.com/)\n", + " - others ?\n", + " \n", + "Those will be your best friends to debug and try different approach for your extensions." + ] + }, + { + "cell_type": "heading", + "level": 3, + "metadata": {}, + "source": [ + "Injecting JS" + ] + }, + { + "cell_type": "heading", + "level": 4, + "metadata": {}, + "source": [ + "using magics" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Above tools can be tedious to edit long javascipt files. Hopefully we provide the `%%javascript` magic. This allows you to quickly inject javascript into the notebook. Still the javascript injected this way will not survive reloading. Hence it is a good tool for testing an refinig a script.\n", + "\n", + "You might see here and there people modifying css and injecting js into notebook by reading file and publishing them into the notebook.\n", + "Not only this often break the flow of the notebook and make the re-execution of the notebook broken, but it also mean that you need to execute those cells on all the notebook every time you need to update the code.\n", + "\n", + "This can still be usefull in some cases, like the `%autosave` magic that allows to control the time between each save. But this can be replaced by a Javascript dropdown menu to select save interval." + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "## you can inspect the autosave code to see what it does.\n", + "%autosave??" + ], + "language": "python", + "metadata": {}, + "outputs": [] + }, + { + "cell_type": "heading", + "level": 4, + "metadata": {}, + "source": [ + "custom.js" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "To inject Javascript we provide an entry point: `custom.js` that allow teh user to execute and load other resources into the notebook.\n", + "Javascript code in `custom.js` will be executed when the notebook app start and can then be used to customise almost anything in the UI and in the behavior of the notebook.\n", + "\n", + "`custom.js` can be found in IPython profile dir, and so you can have different UI modification on a per profile basis, as well as share your modfication with others." + ] + }, + { + "cell_type": "heading", + "level": 5, + "metadata": {}, + "source": [ + "Because we like you...." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You have been provided with an already existing profile folder with this tutorial...\n", + "start the notebook from the root of the tutorial directory with :\n", + "\n", + "```bash\n", + "$ ipython notebook --ProfileDir.location=./profile_euroscipy\n", + "```" + ] + }, + { + "cell_type": "heading", + "level": 5, + "metadata": {}, + "source": [ + "but back to theory" + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "profile_dir = ! ipython locate\n", + "profile_dir = profile_dir[0]\n", + "profile_dir" + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "metadata": {}, + "output_type": "pyout", + "prompt_number": 1, + "text": [ + "'/Users/bussonniermatthias/.ipython'" + ] + } + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "and custom js is in " + ] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "import os.path\n", + "custom_js_path = os.path.join(profile_dir,'profile_default','static','custom','custom.js')" + ], + "language": "python", + "metadata": {}, + "outputs": [] + }, + { + "cell_type": "code", + "collapsed": false, + "input": [ + "# my custom js\n", + "with open(custom_js_path) as f:\n", + " for l in f: \n", + " print l," + ], + "language": "python", + "metadata": {}, + "outputs": [ + { + "output_type": "stream", + "stream": "stdout", + "text": [ + "// we want strict javascript that fails\n", + "// on ambiguous syntax\n", + "\"using strict\";\n", + "\n", + "// notebook loaded is not perfect as it is re-triggerd on\n", + "// revert to checkpoint but this allow extesnsion to be loaded\n", + "// late enough to work.\n", + "$([IPython.events]).on('notebook_loaded.Notebook', function(){\n", + "\n", + "\n", + " /** Use path to js file relative to /static/ dir without leading slash, or\n", + " * js extension.\n", + " * Link directly to file is js extension isa simple file\n", + " *\n", + " * first argument of require is a **list** that can contains several modules if needed.\n", + " **/\n", + "\n", + " //require(['custom/noscroll']);\n", + " // require(['custom/clean_start'])\n", + " // require(['custom/toggle_all_line_number'])\n", + " // require(['custom/gist_it']);\n", + " // require(['custom/autosavetime']);\n", + "\n", + " /**\n", + " * Link to entrypoint if extesnsion is a folder.\n", + " * to be consistent with commonjs module, the entrypoint is main.js\n", + " * here youcan also trigger a custom function on load that will do extra\n", + " * action with the module if needed\n", + " **/\n", + " require(['custom/slidemode/main'])\n", + "\n", + " // require(['custom/autoscroll']);\n", + "\n", + " //require(['custom/css_selector/main'])\n", + " require(['custom/pre_exec_strip']);\n", + " // require(['custom/no_exec_dunder']);\n", + " // load_ext('nbviewer_theme')\n", + "\n", + "\n", + " require(['custom/clippytip/main']);\n", + "\n", + " IPython.toolbar.add_buttons_group([\n", + " {\n", + " 'label' : 'run qtconsole',\n", + " 'icon' : 'icon-paper-clip', // select your icon from http://jqueryui.com/themeroller/\n", + " 'callback': function(){\n", + " IPython.tooltip.remove_and_cancel_tooltip(true)\n", + " $('#tooltip').empty() \n", + " $('#tooltip').attr('style','') \n", + " IPython.tooltip = new IPython.ClippyTip()\n", + " }\n", + " },\n", + " {\n", + " 'label' : 'run qtconsole',\n", + " 'icon' : 'icon-th-large', // select your icon from http://jqueryui.com/themeroller/\n", + " 'callback': function(){\n", + " IPython.tooltip.remove_and_cancel_tooltip(true)\n", + " $('#tooltip').empty() \n", + " $('#tooltip').attr('style','')\n", + " IPython.tooltip = new IPython.Tooltip()\n", + " }\n", + " }\n", + " // add more button here if needed.\n", + " ]);\n", + " //\n", + "\n", + "});\n", + "\n", + "/*\n", + "$([IPython.events]).on('notebook_loaded.Notebook', function(){\n", + " IPython.toolbar.add_buttons_group([\n", + " {\n", + " 'label' : 'run qtconsole',\n", + " 'icon' : 'ui-icon-calculator',\n", + " 'callback': function(){IPython.notebook.kernel.execute('%qtconsole')}\n", + " }\n", + " ]);\n", + "});\n", + "*/\n", + "\n", + "//$([IPython.events]).on('notebook_loaded.Notebook', function(){\n", + "// mobile_preset = []\n", + "// var edit = function(div, cell) {\n", + "// var button_container = $(div);\n", + "// var button = $('
').button({icons:{primary:'ui-icon-pencil'}});\n", + "// button.click(function(){\n", + "// cell.edit()\n", + "// })\n", + "// button_container.append(button);\n", + "// }\n", + "//\n", + "// IPython.CellToolbar.register_callback('mobile.edit',edit);\n", + "// mobile_preset.push('mobile.edit');\n", + "//\n", + "// IPython.CellToolbar.register_preset('Mobile',mobile_preset);\n", + "//});\n" + ] + } + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Note that `custom.js` is ment to be modified by user, when writing a script, you can define it in a separate file and add a line of configuration into `custom.js` that will fetch and execute the file." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "**Warning** : even if modification of `custom.js` take effect immediately after browser refresh (except if browser cache is aggressive), *creating* a file in `static/` directory need a **server restart**." + ] + }, + { + "cell_type": "heading", + "level": 2, + "metadata": {}, + "source": [ + "Exercise :" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + " - Create a `custom.js` in the right location with the following content:\n", + "```javascript\n", + "alert(\"hello world from custom.js\")\n", + "```\n", + "\n", + " - Restart your server and open any notebook.\n", + " - Be greeted by custom.js" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Have a look at [default custom.js](https://github.com/ipython/ipython/blob/1.x/IPython/html/static/custom/custom.js), to see it's content and some more explanation." + ] + }, + { + "cell_type": "heading", + "level": 4, + "metadata": {}, + "source": [ + "For the quick ones : " + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We've seen above that you can change the autosave rate by using a magic. This is typically something I don't want to type everytime, and that I don't like to embed into my workwlow and documents. (reader don't care what my autosave time is), let's build an extension that allow to do it. " + ] + }, + { + "cell_type": "markdown", + "metadata": { + "foo": true + }, + "source": [ + "Create a dropdow elemement in the toolbar (DOM `IPython.toolbar.element`), you will need \n", + "\n", + "- `IPython.notebook.set_autosave_interval(miliseconds)`\n", + "- know that 1min = 60 sec, and 1 sec = 1000 ms" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "```javascript\n", + "\n", + "var label = jQuery('