widget_selectioncontainer.js
324 lines
| 11.7 KiB
| application/javascript
|
JavascriptLexer
Jonathan Frederic
|
r17198 | // Copyright (c) IPython Development Team. | ||
// Distributed under the terms of the Modified BSD License. | ||||
Jonathan Frederic
|
r14366 | |||
Jonathan Frederic
|
r17198 | define([ | ||
"widgets/js/widget", | ||||
Jonathan Frederic
|
r17199 | "base/js/utils", | ||
Jonathan Frederic
|
r17216 | "jquery", | ||
MinRK
|
r17312 | "bootstrap", | ||
Jonathan Frederic
|
r17216 | ], function(widget, utils, $){ | ||
Jonathan Frederic
|
r14366 | |||
Jonathan Frederic
|
r17198 | var AccordionView = widget.DOMWidgetView.extend({ | ||
Jonathan Frederic
|
r19023 | initialize: function(){ | ||
AccordionView.__super__.initialize.apply(this, arguments); | ||||
this.containers = []; | ||||
this.model_containers = {}; | ||||
Jason Grout
|
r19067 | this.children_views = new widget.ViewList(this.add_child_view, this.remove_child_view, this); | ||
Jonathan Frederic
|
r19023 | this.listenTo(this.model, 'change:children', function(model, value) { | ||
this.children_views.update(value); | ||||
}, this); | ||||
}, | ||||
Jonathan Frederic
|
r14284 | render: function(){ | ||
Jonathan Frederic
|
r19176 | /** | ||
* Called when view is rendered. | ||||
*/ | ||||
Jonathan Frederic
|
r17199 | var guid = 'panel-group' + utils.uuid(); | ||
Jonathan Frederic
|
r14455 | this.$el | ||
.attr('id', guid) | ||||
Jonathan Frederic
|
r16913 | .addClass('panel-group'); | ||
Jonathan Frederic
|
r15950 | this.model.on('change:selected_index', function(model, value, options) { | ||
Jonathan Frederic
|
r19773 | this.update_selected_index(options); | ||
Jonathan Frederic
|
r15950 | }, this); | ||
this.model.on('change:_titles', function(model, value, options) { | ||||
Jonathan Frederic
|
r19773 | this.update_titles(options); | ||
Jonathan Frederic
|
r15950 | }, this); | ||
Jonathan Frederic
|
r17179 | this.on('displayed', function() { | ||
Jonathan Frederic
|
r15955 | this.update_titles(); | ||
sylvain.corlay
|
r17354 | }, this); | ||
Jonathan Frederic
|
r19023 | this.children_views.update(this.model.get('children')); | ||
Jason Grout
|
r14503 | }, | ||
Jonathan Frederic
|
r14284 | |||
Jonathan Frederic
|
r19773 | /** | ||
* Update the contents of this view | ||||
* | ||||
* Called when the model is changed. The model may have been | ||||
* changed by another view or by a state update from the back-end. | ||||
*/ | ||||
update: function(options) { | ||||
this.update_titles(); | ||||
this.update_selected_index(options); | ||||
return TabView.__super__.update.apply(this); | ||||
}, | ||||
update_titles: function() { | ||||
Jonathan Frederic
|
r19176 | /** | ||
* Set tab titles | ||||
*/ | ||||
Jonathan Frederic
|
r19773 | var titles = this.model.get('_titles'); | ||
Jonathan Frederic
|
r15950 | var that = this; | ||
_.each(titles, function(title, page_index) { | ||||
var accordian = that.containers[page_index]; | ||||
if (accordian !== undefined) { | ||||
accordian | ||||
Jonathan Frederic
|
r16913 | .find('.panel-heading') | ||
Jonathan Frederic
|
r15950 | .find('.accordion-toggle') | ||
.text(title); | ||||
} | ||||
}); | ||||
}, | ||||
Jonathan Frederic
|
r19773 | update_selected_index: function(options) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Only update the selection if the selection wasn't triggered | ||||
* by the front-end. It must be triggered by the back-end. | ||||
*/ | ||||
Jonathan Frederic
|
r15950 | if (options === undefined || options.updated_view != this) { | ||
Jonathan Frederic
|
r19773 | var old_index = this.model.previous('selected_index'); | ||
var new_index = this.model.get('selected_index'); | ||||
Jonathan Frederic
|
r16913 | this.containers[old_index].find('.panel-collapse').collapse('hide'); | ||
Jonathan Frederic
|
r15950 | if (0 <= new_index && new_index < this.containers.length) { | ||
Jonathan Frederic
|
r16913 | this.containers[new_index].find('.panel-collapse').collapse('show'); | ||
Jonathan Frederic
|
r14370 | } | ||
} | ||||
Jonathan Frederic
|
r14284 | }, | ||
Jason Grout
|
r19067 | remove_child_view: function(view) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Called when a child is removed from children list. | ||||
* TODO: does this handle two different views of the same model as children? | ||||
*/ | ||||
Jason Grout
|
r19067 | var model = view.model; | ||
Jonathan Frederic
|
r14598 | var accordion_group = this.model_containers[model.id]; | ||
this.containers.splice(accordion_group.container_index, 1); | ||||
delete this.model_containers[model.id]; | ||||
accordion_group.remove(); | ||||
}, | ||||
Jonathan Frederic
|
r14284 | |||
Jason Grout
|
r19067 | add_child_view: function(model) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Called when a child is added to children list. | ||||
*/ | ||||
Jonathan Frederic
|
r14284 | var index = this.containers.length; | ||
Jonathan Frederic
|
r17199 | var uuid = utils.uuid(); | ||
Jonathan Frederic
|
r14284 | var accordion_group = $('<div />') | ||
Jonathan Frederic
|
r16913 | .addClass('panel panel-default') | ||
Jonathan Frederic
|
r14284 | .appendTo(this.$el); | ||
var accordion_heading = $('<div />') | ||||
Jonathan Frederic
|
r16913 | .addClass('panel-heading') | ||
Jonathan Frederic
|
r14284 | .appendTo(accordion_group); | ||
Jonathan Frederic
|
r14370 | var that = this; | ||
Jonathan Frederic
|
r14284 | var accordion_toggle = $('<a />') | ||
.addClass('accordion-toggle') | ||||
.attr('data-toggle', 'collapse') | ||||
.attr('data-parent', '#' + this.$el.attr('id')) | ||||
.attr('href', '#' + uuid) | ||||
Jonathan Frederic
|
r14370 | .click(function(evt){ | ||
Jonathan Frederic
|
r14569 | |||
// Calling model.set will trigger all of the other views of the | ||||
// model to update. | ||||
Jonathan Frederic
|
r15950 | that.model.set("selected_index", index, {updated_view: that}); | ||
Jonathan Frederic
|
r14482 | that.touch(); | ||
Jonathan Frederic
|
r14370 | }) | ||
Jonathan Frederic
|
r14663 | .text('Page ' + index) | ||
Jonathan Frederic
|
r14284 | .appendTo(accordion_heading); | ||
var accordion_body = $('<div />', {id: uuid}) | ||||
Jonathan Frederic
|
r16913 | .addClass('panel-collapse collapse') | ||
Jonathan Frederic
|
r14284 | .appendTo(accordion_group); | ||
var accordion_inner = $('<div />') | ||||
Jonathan Frederic
|
r16913 | .addClass('panel-body') | ||
Jonathan Frederic
|
r14284 | .appendTo(accordion_body); | ||
Jonathan Frederic
|
r14598 | var container_index = this.containers.push(accordion_group) - 1; | ||
accordion_group.container_index = container_index; | ||||
this.model_containers[model.id] = accordion_group; | ||||
Thomas Kluyver
|
r18142 | |||
Jonathan Frederic
|
r18883 | var dummy = $('<div/>'); | ||
accordion_inner.append(dummy); | ||||
Jonathan Frederic
|
r18896 | return this.create_child_view(model).then(function(view) { | ||
Jonathan Frederic
|
r18883 | dummy.replaceWith(view.$el); | ||
Thomas Kluyver
|
r18142 | that.update(); | ||
that.update_titles(); | ||||
Jonathan Frederic
|
r16660 | |||
Thomas Kluyver
|
r18142 | // Trigger the displayed event of the child view. | ||
that.after_displayed(function() { | ||||
view.trigger('displayed'); | ||||
}); | ||||
Jonathan Frederic
|
r18896 | return view; | ||
Jason Grout
|
r19080 | }).catch(utils.reject("Couldn't add child view to box", true)); | ||
Jonathan Frederic
|
r14284 | }, | ||
Jonathan Frederic
|
r19023 | |||
remove: function() { | ||||
Jonathan Frederic
|
r19176 | /** | ||
* We remove this widget before removing the children as an optimization | ||||
* we want to remove the entire container from the DOM first before | ||||
* removing each individual child separately. | ||||
*/ | ||||
Jonathan Frederic
|
r19023 | AccordionView.__super__.remove.apply(this, arguments); | ||
this.children_views.remove(); | ||||
}, | ||||
Jonathan Frederic
|
r14284 | }); | ||
Jonathan Frederic
|
r14290 | |||
Jonathan Frederic
|
r14609 | |||
Jonathan Frederic
|
r17198 | var TabView = widget.DOMWidgetView.extend({ | ||
Jonathan Frederic
|
r14507 | initialize: function() { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Public constructor. | ||||
*/ | ||||
Jonathan Frederic
|
r14583 | TabView.__super__.initialize.apply(this, arguments); | ||
Jonathan Frederic
|
r19023 | |||
this.containers = []; | ||||
Jason Grout
|
r19067 | this.children_views = new widget.ViewList(this.add_child_view, this.remove_child_view, this); | ||
Jonathan Frederic
|
r19023 | this.listenTo(this.model, 'change:children', function(model, value) { | ||
this.children_views.update(value); | ||||
}, this); | ||||
Jonathan Frederic
|
r14507 | }, | ||
Jason Grout
|
r14486 | |||
Jonathan Frederic
|
r14290 | render: function(){ | ||
Jonathan Frederic
|
r19176 | /** | ||
* Called when view is rendered. | ||||
*/ | ||||
Jonathan Frederic
|
r17199 | var uuid = 'tabs'+utils.uuid(); | ||
Jonathan Frederic
|
r14290 | this.$tabs = $('<div />', {id: uuid}) | ||
.addClass('nav') | ||||
.addClass('nav-tabs') | ||||
.appendTo(this.$el); | ||||
this.$tab_contents = $('<div />', {id: uuid + 'Content'}) | ||||
.addClass('tab-content') | ||||
.appendTo(this.$el); | ||||
Jonathan Frederic
|
r19023 | this.children_views.update(this.model.get('children')); | ||
Jonathan Frederic
|
r14507 | }, | ||
Jonathan Frederic
|
r14609 | |||
Jonathan Frederic
|
r17724 | update_attr: function(name, value) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Set a css attr of the widget view. | ||||
*/ | ||||
Sylvain Corlay
|
r21464 | if (['padding', 'margin', 'height', 'width'].indexOf(name) !== -1) { | ||
Jonathan Frederic
|
r19366 | this.$el.css(name, value); | ||
} else { | ||||
this.$tabs.css(name, value); | ||||
} | ||||
Jonathan Frederic
|
r17724 | }, | ||
Jason Grout
|
r19067 | remove_child_view: function(view) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Called when a child is removed from children list. | ||||
*/ | ||||
Jonathan Frederic
|
r14598 | this.containers.splice(view.parent_tab.tab_text_index, 1); | ||
view.parent_tab.remove(); | ||||
view.parent_container.remove(); | ||||
view.remove(); | ||||
Jonathan Frederic
|
r14290 | }, | ||
Jason Grout
|
r19067 | add_child_view: function(model) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Called when a child is added to children list. | ||||
*/ | ||||
Jonathan Frederic
|
r14290 | var index = this.containers.length; | ||
Jonathan Frederic
|
r17199 | var uuid = utils.uuid(); | ||
Jonathan Frederic
|
r14290 | |||
var that = this; | ||||
var tab = $('<li />') | ||||
.css('list-style-type', 'none') | ||||
.appendTo(this.$tabs); | ||||
Thomas Kluyver
|
r18142 | |||
Jonathan Frederic
|
r18883 | var tab_text = $('<a />') | ||
.attr('href', '#' + uuid) | ||||
.attr('data-toggle', 'tab') | ||||
.text('Page ' + index) | ||||
.appendTo(tab) | ||||
.click(function (e) { | ||||
// Calling model.set will trigger all of the other views of the | ||||
// model to update. | ||||
that.model.set("selected_index", index, {updated_view: that}); | ||||
that.touch(); | ||||
that.select_page(index); | ||||
}); | ||||
tab.tab_text_index = that.containers.push(tab_text) - 1; | ||||
var dummy = $('<div />'); | ||||
var contents_div = $('<div />', {id: uuid}) | ||||
.addClass('tab-pane') | ||||
.addClass('fade') | ||||
.append(dummy) | ||||
.appendTo(that.$tab_contents); | ||||
Jonathan Frederic
|
r19773 | this.update(); | ||
Jonathan Frederic
|
r18896 | return this.create_child_view(model).then(function(view) { | ||
Jonathan Frederic
|
r18883 | dummy.replaceWith(view.$el); | ||
view.parent_tab = tab; | ||||
Thomas Kluyver
|
r18142 | view.parent_container = contents_div; | ||
// Trigger the displayed event of the child view. | ||||
that.after_displayed(function() { | ||||
view.trigger('displayed'); | ||||
Jonathan Frederic
|
r19773 | that.update(); | ||
Jonathan Frederic
|
r14290 | }); | ||
Jonathan Frederic
|
r18896 | return view; | ||
Jason Grout
|
r19080 | }).catch(utils.reject("Couldn't add child view to box", true)); | ||
Jonathan Frederic
|
r14598 | }, | ||
update: function(options) { | ||||
Jonathan Frederic
|
r19176 | /** | ||
* Update the contents of this view | ||||
* | ||||
* Called when the model is changed. The model may have been | ||||
* changed by another view or by a state update from the back-end. | ||||
*/ | ||||
Jonathan Frederic
|
r19773 | this.update_titles(); | ||
this.update_selected_index(options); | ||||
return TabView.__super__.update.apply(this); | ||||
}, | ||||
Jonathan Frederic
|
r14598 | |||
Jonathan Frederic
|
r19773 | /** | ||
* Updates the tab page titles. | ||||
*/ | ||||
update_titles: function() { | ||||
var titles = this.model.get('_titles'); | ||||
var that = this; | ||||
_.each(titles, function(title, page_index) { | ||||
var tab_text = that.containers[page_index]; | ||||
if (tab_text !== undefined) { | ||||
tab_text.text(title); | ||||
} | ||||
}); | ||||
}, | ||||
/** | ||||
* Updates the tab page titles. | ||||
*/ | ||||
update_selected_index: function(options) { | ||||
if (options === undefined || options.updated_view != this) { | ||||
Jonathan Frederic
|
r14598 | var selected_index = this.model.get('selected_index'); | ||
if (0 <= selected_index && selected_index < this.containers.length) { | ||||
this.select_page(selected_index); | ||||
} | ||||
} | ||||
Jonathan Frederic
|
r14290 | }, | ||
Jonathan Frederic
|
r14307 | |||
select_page: function(index) { | ||||
Jonathan Frederic
|
r19176 | /** | ||
* Select a page. | ||||
*/ | ||||
Jonathan Frederic
|
r14307 | this.$tabs.find('li') | ||
.removeClass('active'); | ||||
this.containers[index].tab('show'); | ||||
}, | ||||
Jonathan Frederic
|
r19023 | |||
remove: function() { | ||||
Jonathan Frederic
|
r19176 | /** | ||
* We remove this widget before removing the children as an optimization | ||||
* we want to remove the entire container from the DOM first before | ||||
* removing each individual child separately. | ||||
*/ | ||||
Jonathan Frederic
|
r19023 | TabView.__super__.remove.apply(this, arguments); | ||
this.children_views.remove(); | ||||
}, | ||||
Jonathan Frederic
|
r14290 | }); | ||
Jonathan Frederic
|
r17198 | |||
return { | ||||
'AccordionView': AccordionView, | ||||
'TabView': TabView, | ||||
}; | ||||
Jonathan Frederic
|
r14290 | }); | ||