##// END OF EJS Templates
rename ipevents to events for consistency
Matthias BUSSONNIER -
Show More
@@ -1,416 +1,416
1 // Copyright (c) IPython Development Team.
1 // Copyright (c) IPython Development Team.
2 // Distributed under the terms of the Modified BSD License.
2 // Distributed under the terms of the Modified BSD License.
3
3
4 define([
4 define([
5 'base/js/namespace',
5 'base/js/namespace',
6 'jquery',
6 'jquery',
7 'base/js/events'
7 'base/js/events'
8 ], function(IPython, $, ipevents) {
8 ], function(IPython, $, events) {
9 "use strict";
9 "use strict";
10
10
11 var CellToolbar = function (options) {
11 var CellToolbar = function (options) {
12 // Constructor
12 // Constructor
13 //
13 //
14 // Parameters:
14 // Parameters:
15 // options: dictionary
15 // options: dictionary
16 // Dictionary of keyword arguments.
16 // Dictionary of keyword arguments.
17 // events: $(Events) instance
17 // events: $(Events) instance
18 // cell: Cell instance
18 // cell: Cell instance
19 // notebook: Notebook instance
19 // notebook: Notebook instance
20 CellToolbar._instances.push(this);
20 CellToolbar._instances.push(this);
21 this.notebook = options.notebook;
21 this.notebook = options.notebook;
22 this.cell = options.cell;
22 this.cell = options.cell;
23 this.create_element();
23 this.create_element();
24 this.rebuild();
24 this.rebuild();
25 return this;
25 return this;
26 };
26 };
27
27
28
28
29 CellToolbar.prototype.create_element = function () {
29 CellToolbar.prototype.create_element = function () {
30 this.inner_element = $('<div/>').addClass('celltoolbar');
30 this.inner_element = $('<div/>').addClass('celltoolbar');
31 this.element = $('<div/>').addClass('ctb_hideshow')
31 this.element = $('<div/>').addClass('ctb_hideshow')
32 .append(this.inner_element);
32 .append(this.inner_element);
33 };
33 };
34
34
35
35
36 // The default css style for the outer celltoolbar div
36 // The default css style for the outer celltoolbar div
37 // (ctb_hideshow) is display: none.
37 // (ctb_hideshow) is display: none.
38 // To show the cell toolbar, *both* of the following conditions must be met:
38 // To show the cell toolbar, *both* of the following conditions must be met:
39 // - A parent container has class `ctb_global_show`
39 // - A parent container has class `ctb_global_show`
40 // - The celltoolbar has the class `ctb_show`
40 // - The celltoolbar has the class `ctb_show`
41 // This allows global show/hide, as well as per-cell show/hide.
41 // This allows global show/hide, as well as per-cell show/hide.
42
42
43 CellToolbar.global_hide = function () {
43 CellToolbar.global_hide = function () {
44 $('body').removeClass('ctb_global_show');
44 $('body').removeClass('ctb_global_show');
45 };
45 };
46
46
47
47
48 CellToolbar.global_show = function () {
48 CellToolbar.global_show = function () {
49 $('body').addClass('ctb_global_show');
49 $('body').addClass('ctb_global_show');
50 };
50 };
51
51
52
52
53 CellToolbar.prototype.hide = function () {
53 CellToolbar.prototype.hide = function () {
54 this.element.removeClass('ctb_show');
54 this.element.removeClass('ctb_show');
55 };
55 };
56
56
57
57
58 CellToolbar.prototype.show = function () {
58 CellToolbar.prototype.show = function () {
59 this.element.addClass('ctb_show');
59 this.element.addClass('ctb_show');
60 };
60 };
61
61
62
62
63 /**
63 /**
64 * Class variable that should contain a dict of all available callback
64 * Class variable that should contain a dict of all available callback
65 * we need to think of wether or not we allow nested namespace
65 * we need to think of wether or not we allow nested namespace
66 * @property _callback_dict
66 * @property _callback_dict
67 * @private
67 * @private
68 * @static
68 * @static
69 * @type Dict
69 * @type Dict
70 */
70 */
71 CellToolbar._callback_dict = {};
71 CellToolbar._callback_dict = {};
72
72
73
73
74 /**
74 /**
75 * Class variable that should contain the reverse order list of the button
75 * Class variable that should contain the reverse order list of the button
76 * to add to the toolbar of each cell
76 * to add to the toolbar of each cell
77 * @property _ui_controls_list
77 * @property _ui_controls_list
78 * @private
78 * @private
79 * @static
79 * @static
80 * @type List
80 * @type List
81 */
81 */
82 CellToolbar._ui_controls_list = [];
82 CellToolbar._ui_controls_list = [];
83
83
84
84
85 /**
85 /**
86 * Class variable that should contain the CellToolbar instances for each
86 * Class variable that should contain the CellToolbar instances for each
87 * cell of the notebook
87 * cell of the notebook
88 *
88 *
89 * @private
89 * @private
90 * @property _instances
90 * @property _instances
91 * @static
91 * @static
92 * @type List
92 * @type List
93 */
93 */
94 CellToolbar._instances = [];
94 CellToolbar._instances = [];
95
95
96
96
97 /**
97 /**
98 * keep a list of all the available presets for the toolbar
98 * keep a list of all the available presets for the toolbar
99 * @private
99 * @private
100 * @property _presets
100 * @property _presets
101 * @static
101 * @static
102 * @type Dict
102 * @type Dict
103 */
103 */
104 CellToolbar._presets = {};
104 CellToolbar._presets = {};
105
105
106
106
107 // this is by design not a prototype.
107 // this is by design not a prototype.
108 /**
108 /**
109 * Register a callback to create an UI element in a cell toolbar.
109 * Register a callback to create an UI element in a cell toolbar.
110 * @method register_callback
110 * @method register_callback
111 * @param name {String} name to use to refer to the callback. It is advised to use a prefix with the name
111 * @param name {String} name to use to refer to the callback. It is advised to use a prefix with the name
112 * for easier sorting and avoid collision
112 * for easier sorting and avoid collision
113 * @param callback {function(div, cell)} callback that will be called to generate the ui element
113 * @param callback {function(div, cell)} callback that will be called to generate the ui element
114 * @param [cell_types] {List of String|undefined} optional list of cell types. If present the UI element
114 * @param [cell_types] {List of String|undefined} optional list of cell types. If present the UI element
115 * will be added only to cells of types in the list.
115 * will be added only to cells of types in the list.
116 *
116 *
117 *
117 *
118 * The callback will receive the following element :
118 * The callback will receive the following element :
119 *
119 *
120 * * a div in which to add element.
120 * * a div in which to add element.
121 * * the cell it is responsible from
121 * * the cell it is responsible from
122 *
122 *
123 * @example
123 * @example
124 *
124 *
125 * Example that create callback for a button that toggle between `true` and `false` label,
125 * Example that create callback for a button that toggle between `true` and `false` label,
126 * with the metadata under the key 'foo' to reflect the status of the button.
126 * with the metadata under the key 'foo' to reflect the status of the button.
127 *
127 *
128 * // first param reference to a DOM div
128 * // first param reference to a DOM div
129 * // second param reference to the cell.
129 * // second param reference to the cell.
130 * var toggle = function(div, cell) {
130 * var toggle = function(div, cell) {
131 * var button_container = $(div)
131 * var button_container = $(div)
132 *
132 *
133 * // let's create a button that show the current value of the metadata
133 * // let's create a button that show the current value of the metadata
134 * var button = $('<div/>').button({label:String(cell.metadata.foo)});
134 * var button = $('<div/>').button({label:String(cell.metadata.foo)});
135 *
135 *
136 * // On click, change the metadata value and update the button label
136 * // On click, change the metadata value and update the button label
137 * button.click(function(){
137 * button.click(function(){
138 * var v = cell.metadata.foo;
138 * var v = cell.metadata.foo;
139 * cell.metadata.foo = !v;
139 * cell.metadata.foo = !v;
140 * button.button("option", "label", String(!v));
140 * button.button("option", "label", String(!v));
141 * })
141 * })
142 *
142 *
143 * // add the button to the DOM div.
143 * // add the button to the DOM div.
144 * button_container.append(button);
144 * button_container.append(button);
145 * }
145 * }
146 *
146 *
147 * // now we register the callback under the name `foo` to give the
147 * // now we register the callback under the name `foo` to give the
148 * // user the ability to use it later
148 * // user the ability to use it later
149 * CellToolbar.register_callback('foo', toggle);
149 * CellToolbar.register_callback('foo', toggle);
150 */
150 */
151 CellToolbar.register_callback = function(name, callback, cell_types) {
151 CellToolbar.register_callback = function(name, callback, cell_types) {
152 // Overwrite if it already exists.
152 // Overwrite if it already exists.
153 CellToolbar._callback_dict[name] = cell_types ? {callback: callback, cell_types: cell_types} : callback;
153 CellToolbar._callback_dict[name] = cell_types ? {callback: callback, cell_types: cell_types} : callback;
154 };
154 };
155
155
156
156
157 /**
157 /**
158 * Register a preset of UI element in a cell toolbar.
158 * Register a preset of UI element in a cell toolbar.
159 * Not supported Yet.
159 * Not supported Yet.
160 * @method register_preset
160 * @method register_preset
161 * @param name {String} name to use to refer to the preset. It is advised to use a prefix with the name
161 * @param name {String} name to use to refer to the preset. It is advised to use a prefix with the name
162 * for easier sorting and avoid collision
162 * for easier sorting and avoid collision
163 * @param preset_list {List of String} reverse order of the button in the toolbar. Each String of the list
163 * @param preset_list {List of String} reverse order of the button in the toolbar. Each String of the list
164 * should correspond to a name of a registerd callback.
164 * should correspond to a name of a registerd callback.
165 *
165 *
166 * @private
166 * @private
167 * @example
167 * @example
168 *
168 *
169 * CellToolbar.register_callback('foo.c1', function(div, cell){...});
169 * CellToolbar.register_callback('foo.c1', function(div, cell){...});
170 * CellToolbar.register_callback('foo.c2', function(div, cell){...});
170 * CellToolbar.register_callback('foo.c2', function(div, cell){...});
171 * CellToolbar.register_callback('foo.c3', function(div, cell){...});
171 * CellToolbar.register_callback('foo.c3', function(div, cell){...});
172 * CellToolbar.register_callback('foo.c4', function(div, cell){...});
172 * CellToolbar.register_callback('foo.c4', function(div, cell){...});
173 * CellToolbar.register_callback('foo.c5', function(div, cell){...});
173 * CellToolbar.register_callback('foo.c5', function(div, cell){...});
174 *
174 *
175 * CellToolbar.register_preset('foo.foo_preset1', ['foo.c1', 'foo.c2', 'foo.c5'])
175 * CellToolbar.register_preset('foo.foo_preset1', ['foo.c1', 'foo.c2', 'foo.c5'])
176 * CellToolbar.register_preset('foo.foo_preset2', ['foo.c4', 'foo.c5'])
176 * CellToolbar.register_preset('foo.foo_preset2', ['foo.c4', 'foo.c5'])
177 */
177 */
178 CellToolbar.register_preset = function(name, preset_list, notebook) {
178 CellToolbar.register_preset = function(name, preset_list, notebook) {
179 CellToolbar._presets[name] = preset_list;
179 CellToolbar._presets[name] = preset_list;
180 ipevents.trigger('preset_added.CellToolbar', {name: name});
180 events.trigger('preset_added.CellToolbar', {name: name});
181 // When "register_callback" is called by a custom extension, it may be executed after notebook is loaded.
181 // When "register_callback" is called by a custom extension, it may be executed after notebook is loaded.
182 // In that case, activate the preset if needed.
182 // In that case, activate the preset if needed.
183 if (notebook && notebook.metadata && notebook.metadata.celltoolbar === name){
183 if (notebook && notebook.metadata && notebook.metadata.celltoolbar === name){
184 CellToolbar.activate_preset(name);
184 CellToolbar.activate_preset(name);
185 }
185 }
186 };
186 };
187
187
188
188
189 /**
189 /**
190 * List the names of the presets that are currently registered.
190 * List the names of the presets that are currently registered.
191 *
191 *
192 * @method list_presets
192 * @method list_presets
193 * @static
193 * @static
194 */
194 */
195 CellToolbar.list_presets = function() {
195 CellToolbar.list_presets = function() {
196 var keys = [];
196 var keys = [];
197 for (var k in CellToolbar._presets) {
197 for (var k in CellToolbar._presets) {
198 keys.push(k);
198 keys.push(k);
199 }
199 }
200 return keys;
200 return keys;
201 };
201 };
202
202
203
203
204 /**
204 /**
205 * Activate an UI preset from `register_preset`
205 * Activate an UI preset from `register_preset`
206 *
206 *
207 * This does not update the selection UI.
207 * This does not update the selection UI.
208 *
208 *
209 * @method activate_preset
209 * @method activate_preset
210 * @param preset_name {String} string corresponding to the preset name
210 * @param preset_name {String} string corresponding to the preset name
211 *
211 *
212 * @static
212 * @static
213 * @private
213 * @private
214 * @example
214 * @example
215 *
215 *
216 * CellToolbar.activate_preset('foo.foo_preset1');
216 * CellToolbar.activate_preset('foo.foo_preset1');
217 */
217 */
218 CellToolbar.activate_preset = function(preset_name){
218 CellToolbar.activate_preset = function(preset_name){
219 var preset = CellToolbar._presets[preset_name];
219 var preset = CellToolbar._presets[preset_name];
220
220
221 if(preset !== undefined){
221 if(preset !== undefined){
222 CellToolbar._ui_controls_list = preset;
222 CellToolbar._ui_controls_list = preset;
223 CellToolbar.rebuild_all();
223 CellToolbar.rebuild_all();
224 }
224 }
225
225
226 ipevents.trigger('preset_activated.CellToolbar', {name: preset_name});
226 events.trigger('preset_activated.CellToolbar', {name: preset_name});
227 };
227 };
228
228
229
229
230 /**
230 /**
231 * This should be called on the class and not on a instance as it will trigger
231 * This should be called on the class and not on a instance as it will trigger
232 * rebuild of all the instances.
232 * rebuild of all the instances.
233 * @method rebuild_all
233 * @method rebuild_all
234 * @static
234 * @static
235 *
235 *
236 */
236 */
237 CellToolbar.rebuild_all = function(){
237 CellToolbar.rebuild_all = function(){
238 for(var i=0; i < CellToolbar._instances.length; i++){
238 for(var i=0; i < CellToolbar._instances.length; i++){
239 CellToolbar._instances[i].rebuild();
239 CellToolbar._instances[i].rebuild();
240 }
240 }
241 };
241 };
242
242
243 /**
243 /**
244 * Rebuild all the button on the toolbar to update its state.
244 * Rebuild all the button on the toolbar to update its state.
245 * @method rebuild
245 * @method rebuild
246 */
246 */
247 CellToolbar.prototype.rebuild = function(){
247 CellToolbar.prototype.rebuild = function(){
248 // strip evrything from the div
248 // strip evrything from the div
249 // which is probably inner_element
249 // which is probably inner_element
250 // or this.element.
250 // or this.element.
251 this.inner_element.empty();
251 this.inner_element.empty();
252 this.ui_controls_list = [];
252 this.ui_controls_list = [];
253
253
254 var callbacks = CellToolbar._callback_dict;
254 var callbacks = CellToolbar._callback_dict;
255 var preset = CellToolbar._ui_controls_list;
255 var preset = CellToolbar._ui_controls_list;
256 // Yes we iterate on the class variable, not the instance one.
256 // Yes we iterate on the class variable, not the instance one.
257 for (var i=0; i < preset.length; i++) {
257 for (var i=0; i < preset.length; i++) {
258 var key = preset[i];
258 var key = preset[i];
259 var callback = callbacks[key];
259 var callback = callbacks[key];
260 if (!callback) continue;
260 if (!callback) continue;
261
261
262 if (typeof callback === 'object') {
262 if (typeof callback === 'object') {
263 if (callback.cell_types.indexOf(this.cell.cell_type) === -1) continue;
263 if (callback.cell_types.indexOf(this.cell.cell_type) === -1) continue;
264 callback = callback.callback;
264 callback = callback.callback;
265 }
265 }
266
266
267 var local_div = $('<div/>').addClass('button_container');
267 var local_div = $('<div/>').addClass('button_container');
268 try {
268 try {
269 callback(local_div, this.cell, this);
269 callback(local_div, this.cell, this);
270 this.ui_controls_list.push(key);
270 this.ui_controls_list.push(key);
271 } catch (e) {
271 } catch (e) {
272 console.log("Error in cell toolbar callback " + key, e);
272 console.log("Error in cell toolbar callback " + key, e);
273 continue;
273 continue;
274 }
274 }
275 // only append if callback succeeded.
275 // only append if callback succeeded.
276 this.inner_element.append(local_div);
276 this.inner_element.append(local_div);
277 }
277 }
278
278
279 // If there are no controls or the cell is a rendered TextCell hide the toolbar.
279 // If there are no controls or the cell is a rendered TextCell hide the toolbar.
280 if (!this.ui_controls_list.length || (this.cell.cell_type != 'code' && this.cell.rendered)) {
280 if (!this.ui_controls_list.length || (this.cell.cell_type != 'code' && this.cell.rendered)) {
281 this.hide();
281 this.hide();
282 } else {
282 } else {
283 this.show();
283 this.show();
284 }
284 }
285 };
285 };
286
286
287
287
288 /**
288 /**
289 */
289 */
290 CellToolbar.utils = {};
290 CellToolbar.utils = {};
291
291
292
292
293 /**
293 /**
294 * A utility function to generate bindings between a checkbox and cell/metadata
294 * A utility function to generate bindings between a checkbox and cell/metadata
295 * @method utils.checkbox_ui_generator
295 * @method utils.checkbox_ui_generator
296 * @static
296 * @static
297 *
297 *
298 * @param name {string} Label in front of the checkbox
298 * @param name {string} Label in front of the checkbox
299 * @param setter {function( cell, newValue )}
299 * @param setter {function( cell, newValue )}
300 * A setter method to set the newValue
300 * A setter method to set the newValue
301 * @param getter {function( cell )}
301 * @param getter {function( cell )}
302 * A getter methods which return the current value.
302 * A getter methods which return the current value.
303 *
303 *
304 * @return callback {function( div, cell )} Callback to be passed to `register_callback`
304 * @return callback {function( div, cell )} Callback to be passed to `register_callback`
305 *
305 *
306 * @example
306 * @example
307 *
307 *
308 * An exmple that bind the subkey `slideshow.isSectionStart` to a checkbox with a `New Slide` label
308 * An exmple that bind the subkey `slideshow.isSectionStart` to a checkbox with a `New Slide` label
309 *
309 *
310 * var newSlide = CellToolbar.utils.checkbox_ui_generator('New Slide',
310 * var newSlide = CellToolbar.utils.checkbox_ui_generator('New Slide',
311 * // setter
311 * // setter
312 * function(cell, value){
312 * function(cell, value){
313 * // we check that the slideshow namespace exist and create it if needed
313 * // we check that the slideshow namespace exist and create it if needed
314 * if (cell.metadata.slideshow == undefined){cell.metadata.slideshow = {}}
314 * if (cell.metadata.slideshow == undefined){cell.metadata.slideshow = {}}
315 * // set the value
315 * // set the value
316 * cell.metadata.slideshow.isSectionStart = value
316 * cell.metadata.slideshow.isSectionStart = value
317 * },
317 * },
318 * //geter
318 * //geter
319 * function(cell){ var ns = cell.metadata.slideshow;
319 * function(cell){ var ns = cell.metadata.slideshow;
320 * // if the slideshow namespace does not exist return `undefined`
320 * // if the slideshow namespace does not exist return `undefined`
321 * // (will be interpreted as `false` by checkbox) otherwise
321 * // (will be interpreted as `false` by checkbox) otherwise
322 * // return the value
322 * // return the value
323 * return (ns == undefined)? undefined: ns.isSectionStart
323 * return (ns == undefined)? undefined: ns.isSectionStart
324 * }
324 * }
325 * );
325 * );
326 *
326 *
327 * CellToolbar.register_callback('newSlide', newSlide);
327 * CellToolbar.register_callback('newSlide', newSlide);
328 *
328 *
329 */
329 */
330 CellToolbar.utils.checkbox_ui_generator = function(name, setter, getter){
330 CellToolbar.utils.checkbox_ui_generator = function(name, setter, getter){
331 return function(div, cell, celltoolbar) {
331 return function(div, cell, celltoolbar) {
332 var button_container = $(div);
332 var button_container = $(div);
333
333
334 var chkb = $('<input/>').attr('type', 'checkbox');
334 var chkb = $('<input/>').attr('type', 'checkbox');
335 var lbl = $('<label/>').append($('<span/>').text(name));
335 var lbl = $('<label/>').append($('<span/>').text(name));
336 lbl.append(chkb);
336 lbl.append(chkb);
337 chkb.attr("checked", getter(cell));
337 chkb.attr("checked", getter(cell));
338
338
339 chkb.click(function(){
339 chkb.click(function(){
340 var v = getter(cell);
340 var v = getter(cell);
341 setter(cell, !v);
341 setter(cell, !v);
342 chkb.attr("checked", !v);
342 chkb.attr("checked", !v);
343 });
343 });
344 button_container.append($('<span/>').append(lbl));
344 button_container.append($('<span/>').append(lbl));
345 };
345 };
346 };
346 };
347
347
348
348
349 /**
349 /**
350 * A utility function to generate bindings between a dropdown list cell
350 * A utility function to generate bindings between a dropdown list cell
351 * @method utils.select_ui_generator
351 * @method utils.select_ui_generator
352 * @static
352 * @static
353 *
353 *
354 * @param list_list {list of sublist} List of sublist of metadata value and name in the dropdown list.
354 * @param list_list {list of sublist} List of sublist of metadata value and name in the dropdown list.
355 * subslit shoud contain 2 element each, first a string that woul be displayed in the dropdown list,
355 * subslit shoud contain 2 element each, first a string that woul be displayed in the dropdown list,
356 * and second the corresponding value to be passed to setter/return by getter. the corresponding value
356 * and second the corresponding value to be passed to setter/return by getter. the corresponding value
357 * should not be "undefined" or behavior can be unexpected.
357 * should not be "undefined" or behavior can be unexpected.
358 * @param setter {function( cell, newValue )}
358 * @param setter {function( cell, newValue )}
359 * A setter method to set the newValue
359 * A setter method to set the newValue
360 * @param getter {function( cell )}
360 * @param getter {function( cell )}
361 * A getter methods which return the current value of the metadata.
361 * A getter methods which return the current value of the metadata.
362 * @param [label=""] {String} optionnal label for the dropdown menu
362 * @param [label=""] {String} optionnal label for the dropdown menu
363 *
363 *
364 * @return callback {function( div, cell )} Callback to be passed to `register_callback`
364 * @return callback {function( div, cell )} Callback to be passed to `register_callback`
365 *
365 *
366 * @example
366 * @example
367 *
367 *
368 * var select_type = CellToolbar.utils.select_ui_generator([
368 * var select_type = CellToolbar.utils.select_ui_generator([
369 * ["<None>" , "None" ],
369 * ["<None>" , "None" ],
370 * ["Header Slide" , "header_slide" ],
370 * ["Header Slide" , "header_slide" ],
371 * ["Slide" , "slide" ],
371 * ["Slide" , "slide" ],
372 * ["Fragment" , "fragment" ],
372 * ["Fragment" , "fragment" ],
373 * ["Skip" , "skip" ],
373 * ["Skip" , "skip" ],
374 * ],
374 * ],
375 * // setter
375 * // setter
376 * function(cell, value){
376 * function(cell, value){
377 * // we check that the slideshow namespace exist and create it if needed
377 * // we check that the slideshow namespace exist and create it if needed
378 * if (cell.metadata.slideshow == undefined){cell.metadata.slideshow = {}}
378 * if (cell.metadata.slideshow == undefined){cell.metadata.slideshow = {}}
379 * // set the value
379 * // set the value
380 * cell.metadata.slideshow.slide_type = value
380 * cell.metadata.slideshow.slide_type = value
381 * },
381 * },
382 * //geter
382 * //geter
383 * function(cell){ var ns = cell.metadata.slideshow;
383 * function(cell){ var ns = cell.metadata.slideshow;
384 * // if the slideshow namespace does not exist return `undefined`
384 * // if the slideshow namespace does not exist return `undefined`
385 * // (will be interpreted as `false` by checkbox) otherwise
385 * // (will be interpreted as `false` by checkbox) otherwise
386 * // return the value
386 * // return the value
387 * return (ns == undefined)? undefined: ns.slide_type
387 * return (ns == undefined)? undefined: ns.slide_type
388 * }
388 * }
389 * CellToolbar.register_callback('slideshow.select', select_type);
389 * CellToolbar.register_callback('slideshow.select', select_type);
390 *
390 *
391 */
391 */
392 CellToolbar.utils.select_ui_generator = function(list_list, setter, getter, label) {
392 CellToolbar.utils.select_ui_generator = function(list_list, setter, getter, label) {
393 label = label || "";
393 label = label || "";
394 return function(div, cell, celltoolbar) {
394 return function(div, cell, celltoolbar) {
395 var button_container = $(div);
395 var button_container = $(div);
396 var lbl = $("<label/>").append($('<span/>').text(label));
396 var lbl = $("<label/>").append($('<span/>').text(label));
397 var select = $('<select/>').addClass('ui-widget ui-widget-content');
397 var select = $('<select/>').addClass('ui-widget ui-widget-content');
398 for(var i=0; i < list_list.length; i++){
398 for(var i=0; i < list_list.length; i++){
399 var opt = $('<option/>')
399 var opt = $('<option/>')
400 .attr('value', list_list[i][1])
400 .attr('value', list_list[i][1])
401 .text(list_list[i][0]);
401 .text(list_list[i][0]);
402 select.append(opt);
402 select.append(opt);
403 }
403 }
404 select.val(getter(cell));
404 select.val(getter(cell));
405 select.change(function(){
405 select.change(function(){
406 setter(cell, select.val());
406 setter(cell, select.val());
407 });
407 });
408 button_container.append($('<span/>').append(lbl).append(select));
408 button_container.append($('<span/>').append(lbl).append(select));
409 };
409 };
410 };
410 };
411
411
412 // Backwards compatability.
412 // Backwards compatability.
413 IPython.CellToolbar = CellToolbar;
413 IPython.CellToolbar = CellToolbar;
414
414
415 return {'CellToolbar': CellToolbar};
415 return {'CellToolbar': CellToolbar};
416 });
416 });
General Comments 0
You need to be logged in to leave comments. Login now