notebooklist.js
891 lines
| 32.8 KiB
| application/javascript
|
JavascriptLexer
Jonathan Frederic
|
r17189 | // Copyright (c) IPython Development Team. | ||
// Distributed under the terms of the Modified BSD License. | ||||
define([ | ||||
'base/js/namespace', | ||||
Jonathan Frederic
|
r17200 | 'jquery', | ||
Jonathan Frederic
|
r17189 | 'base/js/utils', | ||
'base/js/dialog', | ||||
Scott Sanderson
|
r19031 | 'base/js/events', | ||
Jonathan Frederic
|
r19694 | 'base/js/keyboard', | ||
], function(IPython, $, utils, dialog, events, keyboard) { | ||||
MinRK
|
r13063 | "use strict"; | ||
jon
|
r17210 | var NotebookList = function (selector, options) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Constructor | ||||
* | ||||
* Parameters: | ||||
* selector: string | ||||
* options: dictionary | ||||
* Dictionary of keyword arguments. | ||||
* session_list: SessionList instance | ||||
* element_name: string | ||||
* base_url: string | ||||
* notebook_path: string | ||||
* contents: Contents instance | ||||
*/ | ||||
Jonathan Frederic
|
r17189 | var that = this; | ||
jon
|
r17210 | this.session_list = options.session_list; | ||
Paul Ivanov
|
r15454 | // allow code re-use by just changing element_name in kernellist.js | ||
jon
|
r17210 | this.element_name = options.element_name || 'notebook'; | ||
Brian E. Granger
|
r4488 | this.selector = selector; | ||
if (this.selector !== undefined) { | ||||
this.element = $(selector); | ||||
this.style(); | ||||
this.bind_events(); | ||||
} | ||||
MinRK
|
r13063 | this.notebooks_list = []; | ||
this.sessions = {}; | ||||
Jonathan Frederic
|
r17198 | this.base_url = options.base_url || utils.get_body_data("baseUrl"); | ||
this.notebook_path = options.notebook_path || utils.get_body_data("notebookPath"); | ||||
Jeff Hemmelgarn
|
r18643 | this.contents = options.contents; | ||
Jonathan Frederic
|
r17195 | if (this.session_list && this.session_list.events) { | ||
this.session_list.events.on('sessions_loaded.Dashboard', | ||||
function(e, d) { that.sessions_loaded(d); }); | ||||
} | ||||
Jonathan Frederic
|
r20338 | this.selected = []; | ||
Brian E. Granger
|
r4488 | }; | ||
NotebookList.prototype.style = function () { | ||||
Jonathan Frederic
|
r17189 | var prefix = '#' + this.element_name; | ||
Paul Ivanov
|
r15518 | $(prefix + '_toolbar').addClass('list_toolbar'); | ||
$(prefix + '_list_info').addClass('toolbar_info'); | ||||
$(prefix + '_buttons').addClass('toolbar_buttons'); | ||||
$(prefix + '_list_header').addClass('list_header'); | ||||
MinRK
|
r10920 | this.element.addClass("list_container"); | ||
Brian E. Granger
|
r4488 | }; | ||
NotebookList.prototype.bind_events = function () { | ||||
Brian E. Granger
|
r4491 | var that = this; | ||
Paul Ivanov
|
r15456 | $('#refresh_' + this.element_name + '_list').click(function () { | ||
that.load_sessions(); | ||||
Brian Granger
|
r6195 | }); | ||
Brian E. Granger
|
r4491 | this.element.bind('dragover', function () { | ||
return false; | ||||
}); | ||||
Matthias BUSSONNIER
|
r6838 | this.element.bind('drop', function(event){ | ||
Paul Ivanov
|
r15455 | that.handleFilesUpload(event,'drop'); | ||
Brian E. Granger
|
r4491 | return false; | ||
}); | ||||
Jonathan Frederic
|
r19694 | |||
// Bind events for singleton controls. | ||||
if (!NotebookList._bound_singletons) { | ||||
NotebookList._bound_singletons = true; | ||||
$('#new-file').click(function(e) { | ||||
Min RK
|
r20325 | var w = window.open(undefined, IPython._target); | ||
Jonathan Frederic
|
r19699 | that.contents.new_untitled(that.notebook_path || '', {type: 'file', ext: '.txt'}).then(function(data) { | ||
var url = utils.url_join_encode( | ||||
that.base_url, 'edit', data.path | ||||
); | ||||
w.location = url; | ||||
Min RK
|
r20238 | }).catch(function (e) { | ||
w.close(); | ||||
dialog.modal({ | ||||
title: 'Creating File Failed', | ||||
body: $('<div/>') | ||||
.text("An error occurred while creating a new file.") | ||||
.append($('<div/>') | ||||
.addClass('alert alert-danger') | ||||
.text(e.message || e)), | ||||
buttons: { | ||||
OK: {'class': 'btn-primary'} | ||||
} | ||||
}); | ||||
Jonathan Frederic
|
r19699 | }); | ||
Jonathan Frederic
|
r19696 | that.load_sessions(); | ||
Jonathan Frederic
|
r19694 | }); | ||
$('#new-folder').click(function(e) { | ||||
Matthias Bussonnier
|
r19983 | that.contents.new_untitled(that.notebook_path || '', {type: 'directory'}) | ||
.then(function(){ | ||||
that.load_list(); | ||||
Min RK
|
r20238 | }).catch(function (e) { | ||
dialog.modal({ | ||||
title: 'Creating Folder Failed', | ||||
body: $('<div/>') | ||||
.text("An error occurred while creating a new folder.") | ||||
.append($('<div/>') | ||||
.addClass('alert alert-danger') | ||||
.text(e.message || e)), | ||||
buttons: { | ||||
OK: {'class': 'btn-primary'} | ||||
} | ||||
}); | ||||
Matthias Bussonnier
|
r19983 | }); | ||
Min RK
|
r20238 | that.load_sessions(); | ||
Jonathan Frederic
|
r19694 | }); | ||
Jonathan Frederic
|
r20157 | |||
Jonathan Frederic
|
r20313 | // Bind events for action buttons. | ||
Jonathan Frederic
|
r20157 | $('.rename-button').click($.proxy(this.rename_selected, this)); | ||
$('.shutdown-button').click($.proxy(this.shutdown_selected, this)); | ||||
$('.duplicate-button').click($.proxy(this.duplicate_selected, this)); | ||||
$('.delete-button').click($.proxy(this.delete_selected, this)); | ||||
Jonathan Frederic
|
r20313 | |||
// Bind events for selection menu buttons. | ||||
$('#tree-selector .select-all').click($.proxy(this.select_all, this)); | ||||
$('#tree-selector .select-notebooks').click($.proxy(this.select_notebooks, this)); | ||||
$('#tree-selector .select-running-notebooks').click($.proxy(this.select_running_notebooks, this)); | ||||
$('#tree-selector .select-files').click($.proxy(this.select_files, this)); | ||||
$('#tree-selector .select-directories').click($.proxy(this.select_directories, this)); | ||||
Jonathan Frederic
|
r20321 | $('#tree-selector .deselect-all').click($.proxy(this.deselect_all, this)); | ||
Jonathan Frederic
|
r19694 | } | ||
Brian E. Granger
|
r4488 | }; | ||
Paul Ivanov
|
r15455 | NotebookList.prototype.handleFilesUpload = function(event, dropOrForm) { | ||
Matthias BUSSONNIER
|
r6838 | var that = this; | ||
var files; | ||||
if(dropOrForm =='drop'){ | ||||
files = event.originalEvent.dataTransfer.files; | ||||
} else | ||||
{ | ||||
MinRK
|
r13063 | files = event.originalEvent.target.files; | ||
Matthias BUSSONNIER
|
r6838 | } | ||
MinRK
|
r13063 | for (var i = 0; i < files.length; i++) { | ||
var f = files[i]; | ||||
Jonathan Frederic
|
r17198 | var name_and_ext = utils.splitext(f.name); | ||
Matthias BUSSONNIER
|
r13322 | var file_ext = name_and_ext[1]; | ||
MinRK
|
r17536 | |||
var reader = new FileReader(); | ||||
MinRK
|
r13121 | if (file_ext === '.ipynb') { | ||
MinRK
|
r17536 | reader.readAsText(f); | ||
Brian E. Granger
|
r13115 | } else { | ||
MinRK
|
r17536 | // read non-notebook files as binary | ||
reader.readAsArrayBuffer(f); | ||||
MinRK
|
r13063 | } | ||
Jonathan Frederic
|
r20157 | var item = that.new_item(0, true); | ||
MinRK
|
r17536 | item.addClass('new-file'); | ||
Jeffrey Bush
|
r17643 | that.add_name_input(f.name, item, file_ext == '.ipynb' ? 'notebook' : 'file'); | ||
MinRK
|
r17536 | // Store the list item in the reader so we can use it later | ||
// to know which item it belongs to. | ||||
$(reader).data('item', item); | ||||
reader.onload = function (event) { | ||||
var item = $(event.target).data('item'); | ||||
that.add_file_data(event.target.result, item); | ||||
that.add_upload_button(item); | ||||
}; | ||||
Jeffrey Bush
|
r17650 | reader.onerror = function (event) { | ||
var item = $(event.target).data('item'); | ||||
Min RK
|
r18752 | var name = item.data('name'); | ||
Jeffrey Bush
|
r17650 | item.remove(); | ||
dialog.modal({ | ||||
title : 'Failed to read file', | ||||
body : "Failed to read file '" + name + "'", | ||||
buttons : {'OK' : { 'class' : 'btn-primary' }} | ||||
}); | ||||
}; | ||||
Matthias BUSSONNIER
|
r6838 | } | ||
Brian E. Granger
|
r14926 | // Replace the file input form wth a clone of itself. This is required to | ||
// reset the form. Otherwise, if you upload a file, delete it and try to | ||||
// upload it again, the changed event won't fire. | ||||
var form = $('input.fileinput'); | ||||
form.replaceWith(form.clone(true)); | ||||
Matthias BUSSONNIER
|
r6838 | return false; | ||
MinRK
|
r13063 | }; | ||
Brian E. Granger
|
r4488 | |||
Jonathan Frederic
|
r16213 | NotebookList.prototype.clear_list = function (remove_uploads) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Clears the navigation tree. | ||||
* | ||||
* Parameters | ||||
* remove_uploads: bool=False | ||||
* Should upload prompts also be removed from the tree. | ||||
*/ | ||||
Jonathan Frederic
|
r16213 | if (remove_uploads) { | ||
this.element.children('.list_item').remove(); | ||||
} else { | ||||
this.element.children('.list_item:not(.new-file)').remove(); | ||||
} | ||||
Matthias BUSSONNIER
|
r7941 | }; | ||
Brian Granger
|
r6195 | |||
Zachary Sailer
|
r12988 | NotebookList.prototype.load_sessions = function(){ | ||
Jonathan Frederic
|
r17189 | this.session_list.load_sessions(); | ||
Zachary Sailer
|
r12988 | }; | ||
NotebookList.prototype.sessions_loaded = function(data){ | ||||
Paul Ivanov
|
r15479 | this.sessions = data; | ||
Zachary Sailer
|
r12988 | this.load_list(); | ||
}; | ||||
Brian Granger
|
r6195 | |||
Brian E. Granger
|
r4488 | NotebookList.prototype.load_list = function () { | ||
Min RK
|
r18752 | var that = this; | ||
Thomas Kluyver
|
r18827 | this.contents.list_contents(that.notebook_path).then( | ||
$.proxy(this.draw_notebook_list, this), | ||||
function(error) { | ||||
Thomas Kluyver
|
r18678 | that.draw_notebook_list({content: []}, "Server error: " + error.message); | ||
Thomas Kluyver
|
r18653 | } | ||
Thomas Kluyver
|
r18827 | ); | ||
Brian E. Granger
|
r4488 | }; | ||
KesterTong
|
r18636 | /** | ||
* Draw the list of notebooks | ||||
* @method draw_notebook_list | ||||
* @param {Array} list An array of dictionaries representing files or | ||||
Thomas Kluyver
|
r18660 | * directories. | ||
KesterTong
|
r18636 | * @param {String} error_msg An error message | ||
*/ | ||||
Bussonnier Matthias
|
r19479 | |||
var type_order = {'directory':0,'notebook':1,'file':2}; | ||||
KesterTong
|
r18636 | NotebookList.prototype.draw_notebook_list = function (list, error_msg) { | ||
Jonathan Frederic
|
r20302 | // Remember what was selected before the refresh. | ||
var selected_before = this.selected; | ||||
Bussonnier Matthias
|
r19449 | list.content.sort(function(a, b) { | ||
Bussonnier Matthias
|
r19479 | if (type_order[a['type']] < type_order[b['type']]) { | ||
Bussonnier Matthias
|
r19449 | return -1; | ||
} | ||||
Bussonnier Matthias
|
r19479 | if (type_order[a['type']] > type_order[b['type']]) { | ||
Bussonnier Matthias
|
r19449 | return 1; | ||
} | ||||
if (a['name'] < b['name']) { | ||||
return -1; | ||||
} | ||||
if (a['name'] > b['name']) { | ||||
return 1; | ||||
} | ||||
return 0; | ||||
}); | ||||
KesterTong
|
r18636 | var message = error_msg || 'Notebook list empty.'; | ||
Brian E. Granger
|
r15078 | var item = null; | ||
Jeff Hemmelgarn
|
r18640 | var model = null; | ||
var len = list.content.length; | ||||
Matthias BUSSONNIER
|
r6842 | this.clear_list(); | ||
Jeffrey Bush
|
r17651 | var n_uploads = this.element.children('.list_item').length; | ||
MinRK
|
r13063 | if (len === 0) { | ||
MinRK
|
r17526 | item = this.new_item(0); | ||
Brian E. Granger
|
r15078 | var span12 = item.children().first(); | ||
span12.empty(); | ||||
MinRK
|
r15094 | span12.append($('<div style="margin:auto;text-align:center;color:grey"/>').text(message)); | ||
Matthias BUSSONNIER
|
r6857 | } | ||
MinRK
|
r15234 | var path = this.notebook_path; | ||
Jeffrey Bush
|
r17651 | var offset = n_uploads; | ||
Brian E. Granger
|
r15071 | if (path !== '') { | ||
Jonathan Frederic
|
r20157 | item = this.new_item(offset, false); | ||
MinRK
|
r17526 | model = { | ||
type: 'directory', | ||||
name: '..', | ||||
Min RK
|
r18752 | path: utils.url_path_split(path)[0], | ||
MinRK
|
r17526 | }; | ||
this.add_link(model, item); | ||||
Jeffrey Bush
|
r17651 | offset += 1; | ||
Brian E. Granger
|
r15071 | } | ||
Brian E. Granger
|
r4488 | for (var i=0; i<len; i++) { | ||
Jeff Hemmelgarn
|
r18640 | model = list.content[i]; | ||
Jonathan Frederic
|
r20157 | item = this.new_item(i+offset, true); | ||
Jeff Hemmelgarn
|
r18640 | this.add_link(model, item); | ||
MinRK
|
r13063 | } | ||
Scott Sanderson
|
r19031 | // Trigger an event when we've finished drawing the notebook list. | ||
events.trigger('draw_notebook_list.NotebookList'); | ||||
Jonathan Frederic
|
r20302 | |||
// Reselect the items that were selected before. Notify listeners | ||||
// that the selected items may have changed. O(n^2) operation. | ||||
selected_before.forEach(function(item) { | ||||
var list_items = $('.list_item'); | ||||
for (var i=0; i<list_items.length; i++) { | ||||
var $list_item = $(list_items[i]); | ||||
if ($list_item.data('path') == item.path) { | ||||
$list_item.find('input[type=checkbox]').prop('checked', true); | ||||
break; | ||||
} | ||||
} | ||||
}); | ||||
this._selection_changed(); | ||||
Brian E. Granger
|
r4491 | }; | ||
Brian E. Granger
|
r4490 | |||
Brian E. Granger
|
r4491 | |||
Jonathan Frederic
|
r20157 | /** | ||
* Creates a new item. | ||||
* @param {integer} index | ||||
* @param {boolean} [selectable] - tristate, undefined: don't draw checkbox, | ||||
* false: don't draw checkbox but pad | ||||
* where it should be, true: draw checkbox. | ||||
* @return {JQuery} row | ||||
*/ | ||||
NotebookList.prototype.new_item = function (index, selectable) { | ||||
var row = $('<div/>') | ||||
.addClass("list_item") | ||||
.addClass("row"); | ||||
var item = $("<div/>") | ||||
.addClass("col-md-12") | ||||
.appendTo(row); | ||||
var checkbox; | ||||
if (selectable !== undefined) { | ||||
checkbox = $('<input/>') | ||||
.attr('type', 'checkbox') | ||||
Jonathan Frederic
|
r20163 | .attr('title', 'Click here to rename, delete, etc.') | ||
Jonathan Frederic
|
r20157 | .appendTo(item); | ||
} | ||||
$('<i/>') | ||||
.addClass('item_icon') | ||||
.appendTo(item); | ||||
var link = $("<a/>") | ||||
.addClass("item_link") | ||||
.appendTo(item); | ||||
$("<span/>") | ||||
.addClass("item_name") | ||||
.appendTo(link); | ||||
if (selectable === false) { | ||||
checkbox.css('visibility', 'hidden'); | ||||
} else if (selectable === true) { | ||||
var that = this; | ||||
row.click(function(e) { | ||||
Mathieu
|
r20358 | // toggle checkbox only if the click doesn't come from the checkbox or the link | ||
if (!$(e.target).is('span[class=item_name]') && !$(e.target).is('input[type=checkbox]')) { | ||||
checkbox.prop('checked', !checkbox.prop('checked')); | ||||
} | ||||
Jonathan Frederic
|
r20157 | that._selection_changed(); | ||
}); | ||||
} | ||||
var buttons = $('<div/>') | ||||
.addClass("item_buttons pull-right") | ||||
.appendTo(item); | ||||
Jonathan Frederic
|
r20161 | $('<div/>') | ||
.addClass('running-indicator') | ||||
.text('Running') | ||||
.css('visibility', 'hidden') | ||||
Jonathan Frederic
|
r20157 | .appendTo(buttons); | ||
MinRK
|
r10891 | |||
Brian E. Granger
|
r4491 | if (index === -1) { | ||
Jonathan Frederic
|
r20157 | this.element.append(row); | ||
Brian E. Granger
|
r4491 | } else { | ||
Jonathan Frederic
|
r20157 | this.element.children().eq(index).after(row); | ||
Brian E. Granger
|
r4488 | } | ||
Jonathan Frederic
|
r20157 | return row; | ||
Brian E. Granger
|
r4491 | }; | ||
MinRK
|
r17526 | NotebookList.icons = { | ||
directory: 'folder_icon', | ||||
notebook: 'notebook_icon', | ||||
file: 'file_icon', | ||||
}; | ||||
NotebookList.uri_prefixes = { | ||||
directory: 'tree', | ||||
notebook: 'notebooks', | ||||
Min RK
|
r19307 | file: 'edit', | ||
MinRK
|
r17526 | }; | ||
Jonathan Frederic
|
r20241 | /** | ||
Jonathan Frederic
|
r20313 | * Select all of the items in the tree. | ||
*/ | ||||
NotebookList.prototype.select_all = function() { | ||||
$('.list_item input[type=checkbox]').each(function(index, item) { | ||||
$(item).prop('checked', true); | ||||
}); | ||||
this._selection_changed(); | ||||
}; | ||||
/** | ||||
* Select all of the notebooks in the tree. | ||||
*/ | ||||
NotebookList.prototype.select_notebooks = function() { | ||||
Jonathan Frederic
|
r20321 | this.deselect_all(); | ||
Jonathan Frederic
|
r20313 | $('.list_item').each(function(index, item) { | ||
if ($(item).data('type') === 'notebook') { | ||||
$(item).find('input[type=checkbox]').prop('checked', true); | ||||
} | ||||
}); | ||||
this._selection_changed(); | ||||
}; | ||||
/** | ||||
* Select all of the running notebooks in the tree. | ||||
*/ | ||||
NotebookList.prototype.select_running_notebooks = function() { | ||||
Jonathan Frederic
|
r20321 | this.deselect_all(); | ||
Jonathan Frederic
|
r20313 | var that = this; | ||
$('.list_item').each(function(index, item) { | ||||
if ($(item).data('type') === 'notebook' && that.sessions[$(item).data('path')] !== undefined) { | ||||
$(item).find('input[type=checkbox]').prop('checked', true); | ||||
} | ||||
}); | ||||
this._selection_changed(); | ||||
}; | ||||
/** | ||||
* Select all of the files in the tree. | ||||
*/ | ||||
NotebookList.prototype.select_files = function() { | ||||
Jonathan Frederic
|
r20321 | this.deselect_all(); | ||
Jonathan Frederic
|
r20313 | $('.list_item').each(function(index, item) { | ||
if ($(item).data('type') === 'file') { | ||||
$(item).find('input[type=checkbox]').prop('checked', true); | ||||
} | ||||
}); | ||||
this._selection_changed(); | ||||
}; | ||||
/** | ||||
* Select all of the directories in the tree. | ||||
*/ | ||||
NotebookList.prototype.select_directories = function() { | ||||
Jonathan Frederic
|
r20321 | this.deselect_all(); | ||
Jonathan Frederic
|
r20313 | $('.list_item').each(function(index, item) { | ||
if ($(item).data('type') === 'directory') { | ||||
$(item).find('input[type=checkbox]').prop('checked', true); | ||||
} | ||||
}); | ||||
this._selection_changed(); | ||||
}; | ||||
/** | ||||
* Unselect everything selected in the tree. | ||||
*/ | ||||
Jonathan Frederic
|
r20321 | NotebookList.prototype.deselect_all = function() { | ||
Jonathan Frederic
|
r20313 | $('.list_item input[type=checkbox]').each(function(index, item) { | ||
$(item).prop('checked', false); | ||||
}); | ||||
this._selection_changed(); | ||||
}; | ||||
/** | ||||
Jonathan Frederic
|
r20241 | * Handles when any row selector checkbox is toggled. | ||
*/ | ||||
Jonathan Frederic
|
r20157 | NotebookList.prototype._selection_changed = function() { | ||
Jonathan Frederic
|
r20241 | // Use a JQuery selector to find each row with a checked checkbox. If | ||
// we decide to add more checkboxes in the future, this code will need | ||||
// to be changed to distinguish which checkbox is the row selector. | ||||
Jonathan Frederic
|
r20157 | var selected = []; | ||
Jonathan Frederic
|
r20160 | var has_running_notebook = false; | ||
Jonathan Frederic
|
r20157 | var has_directory = false; | ||
Jonathan Frederic
|
r20165 | var has_file = false; | ||
Jonathan Frederic
|
r20162 | var that = this; | ||
Jonathan Frederic
|
r20313 | var checked = 0; | ||
Jonathan Frederic
|
r20157 | $('.list_item :checked').each(function(index, item) { | ||
var parent = $(item).parent().parent(); | ||||
Jonathan Frederic
|
r20313 | // If the item doesn't have an upload button and it's not the | ||
// breadcrumbs, it can be selected. Breadcrumbs path == ''. | ||||
if (parent.find('.upload_button').length === 0 && parent.data('path') !== '') { | ||||
checked++; | ||||
Jonathan Frederic
|
r20239 | selected.push({ | ||
name: parent.data('name'), | ||||
path: parent.data('path'), | ||||
type: parent.data('type') | ||||
}); | ||||
Jonathan Frederic
|
r20157 | |||
Jonathan Frederic
|
r20241 | // Set flags according to what is selected. Flags are later | ||
// used to decide which action buttons are visible. | ||||
Jonathan Frederic
|
r20239 | has_running_notebook = has_running_notebook || | ||
(parent.data('type') == 'notebook' && that.sessions[parent.data('path')] !== undefined); | ||||
has_file = has_file || parent.data('type') == 'file'; | ||||
has_directory = has_directory || parent.data('type') == 'directory'; | ||||
} | ||||
Jonathan Frederic
|
r20157 | }); | ||
this.selected = selected; | ||||
// Rename is only visible when one item is selected. | ||||
if (selected.length==1) { | ||||
$('.rename-button').css('display', 'inline-block'); | ||||
} else { | ||||
$('.rename-button').css('display', 'none'); | ||||
} | ||||
Jonathan Frederic
|
r20241 | // Shutdown is only visible when one or more notebooks running notebooks | ||
// are selected and no non-notebook items are selected. | ||||
Jonathan Frederic
|
r20165 | if (has_running_notebook && !(has_file || has_directory)) { | ||
Jonathan Frederic
|
r20157 | $('.shutdown-button').css('display', 'inline-block'); | ||
} else { | ||||
$('.shutdown-button').css('display', 'none'); | ||||
} | ||||
Jonathan Frederic
|
r20241 | // Duplicate isn't visible when a directory is selected. | ||
Jonathan Frederic
|
r20157 | if (selected.length > 0 && !has_directory) { | ||
$('.duplicate-button').css('display', 'inline-block'); | ||||
} else { | ||||
$('.duplicate-button').css('display', 'none'); | ||||
} | ||||
// Delete is visible if one or more items are selected. | ||||
if (selected.length > 0) { | ||||
$('.delete-button').css('display', 'inline-block'); | ||||
} else { | ||||
$('.delete-button').css('display', 'none'); | ||||
} | ||||
Jonathan Frederic
|
r20313 | |||
// If all of the items are selected, show the selector as checked. If | ||||
// some of the items are selected, show it as checked. Otherwise, | ||||
// uncheck it. | ||||
Jonathan Frederic
|
r20320 | var total = 0; | ||
$('.list_item input[type=checkbox]').each(function(index, item) { | ||||
var parent = $(item).parent().parent(); | ||||
// If the item doesn't have an upload button and it's not the | ||||
// breadcrumbs, it can be selected. Breadcrumbs path == ''. | ||||
if (parent.find('.upload_button').length === 0 && parent.data('path') !== '') { | ||||
total++; | ||||
} | ||||
}); | ||||
Jonathan Frederic
|
r20313 | if (checked === 0) { | ||
Jonathan Frederic
|
r20320 | $('#tree-selector input[type=checkbox]')[0].indeterminate = false; | ||
Jonathan Frederic
|
r20313 | $('#tree-selector input[type=checkbox]').prop('checked', false); | ||
Jonathan Frederic
|
r20320 | } else if (checked === total) { | ||
$('#tree-selector input[type=checkbox]')[0].indeterminate = false; | ||||
Jonathan Frederic
|
r20313 | $('#tree-selector input[type=checkbox]').prop('checked', true); | ||
Jonathan Frederic
|
r20320 | } else { | ||
$('#tree-selector input[type=checkbox]').prop('checked', false); | ||||
$('#tree-selector input[type=checkbox]')[0].indeterminate = true; | ||||
Jonathan Frederic
|
r20313 | } | ||
Jonathan Frederic
|
r20157 | }; | ||
MinRK
|
r17526 | |||
NotebookList.prototype.add_link = function (model, item) { | ||||
var path = model.path, | ||||
name = model.name; | ||||
Mathieu
|
r20235 | var running = (model.type == 'notebook' && this.sessions[path] !== undefined); | ||
Brian E. Granger
|
r15071 | item.data('name', name); | ||
item.data('path', path); | ||||
Jonathan Frederic
|
r20157 | item.data('type', model.type); | ||
Brian E. Granger
|
r15071 | item.find(".item_name").text(name); | ||
MinRK
|
r17526 | var icon = NotebookList.icons[model.type]; | ||
Mathieu
|
r20235 | if (running) { | ||
icon = 'running_' + icon; | ||||
} | ||||
MinRK
|
r17526 | var uri_prefix = NotebookList.uri_prefixes[model.type]; | ||
item.find(".item_icon").addClass(icon).addClass('icon-fixed-width'); | ||||
MinRK
|
r17535 | var link = item.find("a.item_link") | ||
Brian E. Granger
|
r15071 | .attr('href', | ||
Jonathan Frederic
|
r17198 | utils.url_join_encode( | ||
MinRK
|
r15238 | this.base_url, | ||
MinRK
|
r17526 | uri_prefix, | ||
Min RK
|
r18752 | path | ||
Brian E. Granger
|
r15071 | ) | ||
); | ||||
Jonathan Frederic
|
r20156 | |||
Jonathan Frederic
|
r20161 | item.find(".item_buttons .running-indicator").css('visibility', running ? '' : 'hidden'); | ||
Jonathan Frederic
|
r20156 | |||
MinRK
|
r17535 | // directory nav doesn't open new tabs | ||
// files, notebooks do | ||||
if (model.type !== "directory") { | ||||
Min RK
|
r20326 | link.attr('target',IPython._target); | ||
MinRK
|
r17535 | } | ||
Brian E. Granger
|
r15071 | }; | ||
Jeffrey Bush
|
r17643 | NotebookList.prototype.add_name_input = function (name, item, icon_type) { | ||
MinRK
|
r17526 | item.data('name', name); | ||
Jeffrey Bush
|
r17643 | item.find(".item_icon").addClass(NotebookList.icons[icon_type]).addClass('icon-fixed-width'); | ||
MinRK
|
r10919 | item.find(".item_name").empty().append( | ||
MinRK
|
r10896 | $('<input/>') | ||
MinRK
|
r17536 | .addClass("filename_input") | ||
.attr('value', name) | ||||
MinRK
|
r10896 | .attr('size', '30') | ||
.attr('type', 'text') | ||||
Jeffrey Bush
|
r17646 | .keyup(function(event){ | ||
if(event.keyCode == 13){item.find('.upload_button').click();} | ||||
else if(event.keyCode == 27){item.remove();} | ||||
}) | ||||
Brian E. Granger
|
r4491 | ); | ||
}; | ||||
MinRK
|
r17536 | NotebookList.prototype.add_file_data = function (data, item) { | ||
item.data('filedata', data); | ||||
Brian E. Granger
|
r4491 | }; | ||
Jonathan Frederic
|
r20157 | NotebookList.prototype.shutdown_selected = function() { | ||
Matthias BUSSONNIER
|
r6842 | var that = this; | ||
Jonathan Frederic
|
r20164 | this.selected.forEach(function(item) { | ||
if (item.type == 'notebook') { | ||||
that.shutdown_notebook(item.path); | ||||
} | ||||
}); | ||||
}; | ||||
NotebookList.prototype.shutdown_notebook = function(path) { | ||||
var that = this; | ||||
Jonathan Frederic
|
r20157 | var settings = { | ||
processData : false, | ||||
cache : false, | ||||
type : "DELETE", | ||||
dataType : "json", | ||||
success : function () { | ||||
that.load_sessions(); | ||||
}, | ||||
error : utils.log_ajax_error, | ||||
}; | ||||
Jonathan Frederic
|
r20164 | var session = this.sessions[path]; | ||
if (session) { | ||||
var url = utils.url_join_encode( | ||||
this.base_url, | ||||
'api/sessions', | ||||
session | ||||
); | ||||
$.ajax(url, settings); | ||||
} | ||||
Min RK
|
r20238 | }; | ||
David Neto
|
r18927 | |||
Jonathan Frederic
|
r20157 | NotebookList.prototype.rename_selected = function() { | ||
if (this.selected.length != 1) return; | ||||
Jonathan Frederic
|
r20156 | |||
Jonathan Frederic
|
r20157 | var that = this; | ||
var path = this.selected[0].path; | ||||
var input = $('<input/>').attr('type','text').attr('size','25').addClass('form-control') | ||||
.val(path); | ||||
var dialog_body = $('<div/>').append( | ||||
$("<p/>").addClass("rename-message") | ||||
.text('Enter a new directory name:') | ||||
).append( | ||||
$("<br/>") | ||||
).append(input); | ||||
var d = dialog.modal({ | ||||
title : "Rename directory", | ||||
body : dialog_body, | ||||
buttons : { | ||||
OK : { | ||||
class: "btn-primary", | ||||
click: function() { | ||||
that.contents.rename(path, input.val()).then(function() { | ||||
that.load_list(); | ||||
}).catch(function(e) { | ||||
dialog.modal({ | ||||
Min RK
|
r20238 | title: "Rename Failed", | ||
body: $('<div/>') | ||||
Jonathan Frederic
|
r20157 | .text("An error occurred while renaming \"" + path + "\" to \"" + input.val() + "\".") | ||
Min RK
|
r20238 | .append($('<div/>') | ||
.addClass('alert alert-danger') | ||||
.text(e.message || e)), | ||||
buttons: { | ||||
OK: {'class': 'btn-primary'} | ||||
Jonathan Frederic
|
r20157 | } | ||
}); | ||||
}); | ||||
} | ||||
}, | ||||
Cancel : {} | ||||
}, | ||||
open : function () { | ||||
// Upon ENTER, click the OK button. | ||||
input.keydown(function (event) { | ||||
if (event.which === keyboard.keycodes.enter) { | ||||
d.find('.btn-primary').first().click(); | ||||
return false; | ||||
} | ||||
}); | ||||
input.focus().select(); | ||||
} | ||||
}); | ||||
Jonathan Frederic
|
r20156 | }; | ||
Jonathan Frederic
|
r20157 | NotebookList.prototype.delete_selected = function() { | ||
var message; | ||||
if (this.selected.length == 1) { | ||||
message = 'Are you sure you want to permanently delete: ' + this.selected[0].name + '?'; | ||||
} else { | ||||
Jonathan Frederic
|
r20225 | message = 'Are you sure you want to permanently delete the ' + this.selected.length + ' files/folders selected?'; | ||
Jonathan Frederic
|
r20157 | } | ||
var that = this; | ||||
dialog.modal({ | ||||
title : "Delete", | ||||
body : message, | ||||
buttons : { | ||||
Delete : { | ||||
class: "btn-danger", | ||||
click: function() { | ||||
Jonathan Frederic
|
r20159 | // Shutdown any/all selected notebooks before deleting | ||
// the files. | ||||
that.shutdown_selected(); | ||||
// Delete selected. | ||||
Jonathan Frederic
|
r20157 | that.selected.forEach(function(item) { | ||
that.contents.delete(item.path).then(function() { | ||||
that.notebook_deleted(item.path); | ||||
Jonathan Frederic
|
r20156 | }).catch(function(e) { | ||
dialog.modal({ | ||||
Min RK
|
r20238 | title: "Delete Failed", | ||
body: $('<div/>') | ||||
Jonathan Frederic
|
r20157 | .text("An error occurred while deleting \"" + item.path + "\".") | ||
Min RK
|
r20238 | .append($('<div/>') | ||
.addClass('alert alert-danger') | ||||
.text(e.message || e)), | ||||
buttons: { | ||||
OK: {'class': 'btn-primary'} | ||||
Jonathan Frederic
|
r20156 | } | ||
Min RK
|
r18930 | }); | ||
Jonathan Frederic
|
r20156 | }); | ||
Jonathan Frederic
|
r20157 | }); | ||
} | ||||
Jonathan Frederic
|
r20156 | }, | ||
Jonathan Frederic
|
r20157 | Cancel : {} | ||
} | ||||
}); | ||||
Matthias BUSSONNIER
|
r6842 | }; | ||
Jonathan Frederic
|
r20157 | NotebookList.prototype.duplicate_selected = function() { | ||
var message; | ||||
if (this.selected.length == 1) { | ||||
message = 'Are you sure you want to duplicate: ' + this.selected[0].name + '?'; | ||||
} else { | ||||
message = 'Are you sure you want to duplicate the ' + this.selected.length + ' files selected?'; | ||||
} | ||||
var that = this; | ||||
dialog.modal({ | ||||
title : "Delete", | ||||
body : message, | ||||
buttons : { | ||||
Duplicate : { | ||||
class: "btn-primary", | ||||
click: function() { | ||||
that.selected.forEach(function(item) { | ||||
that.contents.copy(item.path, that.notebook_path).then(function () { | ||||
that.load_list(); | ||||
Jonathan Frederic
|
r20156 | }).catch(function(e) { | ||
dialog.modal({ | ||||
Min RK
|
r20238 | title: "Delete Failed", | ||
body: $('<div/>') | ||||
.text("An error occurred while deleting \"" + item.path + "\".") | ||||
.append($('<div/>') | ||||
.addClass('alert alert-danger') | ||||
.text(e.message || e)), | ||||
buttons: { | ||||
OK: {'class': 'btn-primary'} | ||||
Thomas Kluyver
|
r18656 | } | ||
Jonathan Frederic
|
r20156 | }); | ||
}); | ||||
Jonathan Frederic
|
r20157 | }); | ||
} | ||||
}, | ||||
Cancel : {} | ||||
} | ||||
}); | ||||
Brian E. Granger
|
r4491 | }; | ||
Min RK
|
r18752 | NotebookList.prototype.notebook_deleted = function(path) { | ||
Jonathan Frederic
|
r19176 | /** | ||
* Remove the deleted notebook. | ||||
*/ | ||||
Min RK
|
r20184 | var that = this; | ||
Thomas Kluyver
|
r18772 | $( ":data(path)" ).each(function() { | ||
Min RK
|
r18752 | var element = $(this); | ||
Min RK
|
r20184 | if (element.data("path") === path) { | ||
Thomas Kluyver
|
r18655 | element.remove(); | ||
Scott Sanderson
|
r19031 | events.trigger('notebook_deleted.NotebookList'); | ||
Min RK
|
r20184 | that._selection_changed(); | ||
Thomas Kluyver
|
r18655 | } | ||
}); | ||||
Min RK
|
r18752 | }; | ||
Thomas Kluyver
|
r18655 | |||
Brian E. Granger
|
r4491 | |||
Min RK
|
r18752 | NotebookList.prototype.add_upload_button = function (item) { | ||
Brian E. Granger
|
r4491 | var that = this; | ||
MinRK
|
r10896 | var upload_button = $('<button/>').text("Upload") | ||
Jonathan Frederic
|
r16914 | .addClass('btn btn-primary btn-xs upload_button') | ||
MinRK
|
r10896 | .click(function (e) { | ||
MinRK
|
r17536 | var filename = item.find('.item_name > input').val(); | ||
Min RK
|
r18752 | var path = utils.url_path_join(that.notebook_path, filename); | ||
MinRK
|
r17536 | var filedata = item.data('filedata'); | ||
var format = 'text'; | ||||
Jeffrey Bush
|
r17650 | if (filename.length === 0 || filename[0] === '.') { | ||
dialog.modal({ | ||||
title : 'Invalid file name', | ||||
body : "File names must be at least one character and not start with a dot", | ||||
buttons : {'OK' : { 'class' : 'btn-primary' }} | ||||
}); | ||||
return false; | ||||
} | ||||
MinRK
|
r17536 | if (filedata instanceof ArrayBuffer) { | ||
// base64-encode binary file data | ||||
var bytes = ''; | ||||
var buf = new Uint8Array(filedata); | ||||
var nbytes = buf.byteLength; | ||||
for (var i=0; i<nbytes; i++) { | ||||
bytes += String.fromCharCode(buf[i]); | ||||
} | ||||
filedata = btoa(bytes); | ||||
format = 'base64'; | ||||
} | ||||
Min RK
|
r18752 | var model = {}; | ||
MinRK
|
r17536 | |||
var name_and_ext = utils.splitext(filename); | ||||
var file_ext = name_and_ext[1]; | ||||
var content_type; | ||||
if (file_ext === '.ipynb') { | ||||
model.type = 'notebook'; | ||||
model.format = 'json'; | ||||
MinRK
|
r17537 | try { | ||
model.content = JSON.parse(filedata); | ||||
} catch (e) { | ||||
dialog.modal({ | ||||
title : 'Cannot upload invalid Notebook', | ||||
body : "The error was: " + e, | ||||
buttons : {'OK' : { | ||||
'class' : 'btn-primary', | ||||
click: function () { | ||||
item.remove(); | ||||
} | ||||
}} | ||||
}); | ||||
Jeffrey Bush
|
r17650 | return false; | ||
MinRK
|
r17537 | } | ||
MinRK
|
r17536 | content_type = 'application/json'; | ||
} else { | ||||
model.type = 'file'; | ||||
model.format = format; | ||||
model.content = filedata; | ||||
content_type = 'application/octet-stream'; | ||||
} | ||||
Min RK
|
r18752 | filedata = item.data('filedata'); | ||
MinRK
|
r17536 | |||
Thomas Kluyver
|
r18829 | var on_success = function () { | ||
item.removeClass('new-file'); | ||||
that.add_link(model, item); | ||||
that.session_list.load_sessions(); | ||||
Brian E. Granger
|
r4491 | }; | ||
Jeffrey Bush
|
r17648 | |||
var exists = false; | ||||
$.each(that.element.find('.list_item:not(.new-file)'), function(k,v){ | ||||
if ($(v).data('name') === filename) { exists = true; return false; } | ||||
}); | ||||
Min RK
|
r18752 | |||
Jeffrey Bush
|
r17648 | if (exists) { | ||
dialog.modal({ | ||||
title : "Replace file", | ||||
body : 'There is already a file named ' + filename + ', do you want to replace it?', | ||||
buttons : { | ||||
Overwrite : { | ||||
class: "btn-danger", | ||||
Min RK
|
r18752 | click: function () { | ||
Thomas Kluyver
|
r18829 | that.contents.save(path, model).then(on_success); | ||
} | ||||
Jeffrey Bush
|
r17648 | }, | ||
Cancel : { | ||||
click: function() { item.remove(); } | ||||
} | ||||
} | ||||
}); | ||||
} else { | ||||
Thomas Kluyver
|
r18829 | that.contents.save(path, model).then(on_success); | ||
Jeffrey Bush
|
r17648 | } | ||
MinRK
|
r10891 | return false; | ||
Brian E. Granger
|
r4491 | }); | ||
MinRK
|
r10896 | var cancel_button = $('<button/>').text("Cancel") | ||
Jonathan Frederic
|
r16913 | .addClass("btn btn-default btn-xs") | ||
MinRK
|
r10896 | .click(function (e) { | ||
Brian E. Granger
|
r4491 | item.remove(); | ||
MinRK
|
r10891 | return false; | ||
Brian E. Granger
|
r4491 | }); | ||
MinRK
|
r10896 | item.find(".item_buttons").empty() | ||
MinRK
|
r10891 | .append(upload_button) | ||
.append(cancel_button); | ||||
Brian E. Granger
|
r4488 | }; | ||
Min RK
|
r18752 | // Backwards compatability. | ||
Brian E. Granger
|
r4488 | IPython.NotebookList = NotebookList; | ||
Jonathan Frederic
|
r17201 | return {'NotebookList': NotebookList}; | ||
Jonathan Frederic
|
r17189 | }); | ||