notebooklist.js
448 lines
| 15.9 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', | ||||
Jonathan Frederic
|
r17202 | ], function(IPython, $, utils, dialog) { | ||
MinRK
|
r13063 | "use strict"; | ||
jon
|
r17210 | var NotebookList = function (selector, options) { | ||
jon
|
r17211 | // Constructor | ||
// | ||||
// Parameters: | ||||
// selector: string | ||||
// options: dictionary | ||||
// Dictionary of keyword arguments. | ||||
// session_list: SessionList instance | ||||
// element_name: string | ||||
// base_url: string | ||||
// notebook_path: string | ||||
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"); | ||||
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); }); | ||||
} | ||||
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; | ||
}); | ||||
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]; | ||||
Matthias BUSSONNIER
|
r6838 | var reader = new FileReader(); | ||
reader.readAsText(f); | ||||
Jonathan Frederic
|
r17198 | var name_and_ext = utils.splitext(f.name); | ||
Matthias BUSSONNIER
|
r13322 | var file_ext = name_and_ext[1]; | ||
MinRK
|
r13121 | if (file_ext === '.ipynb') { | ||
Matthias BUSSONNIER
|
r6838 | var item = that.new_notebook_item(0); | ||
Jonathan Frederic
|
r16213 | item.addClass('new-file'); | ||
MinRK
|
r15094 | that.add_name_input(f.name, item); | ||
Matthias BUSSONNIER
|
r6838 | // Store the notebook 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 nbitem = $(event.target).data('item'); | ||||
that.add_notebook_data(event.target.result, nbitem); | ||||
that.add_upload_button(nbitem); | ||||
}; | ||||
Brian E. Granger
|
r13115 | } else { | ||
Jonathan Frederic
|
r17203 | var dialog_body = 'Uploaded notebooks must be .ipynb files'; | ||
Jonathan Frederic
|
r17202 | dialog.modal({ | ||
Brian E. Granger
|
r13115 | title : 'Invalid file type', | ||
Jonathan Frederic
|
r17203 | body : dialog_body, | ||
Brian E. Granger
|
r13115 | buttons : {'OK' : {'class' : 'btn-primary'}} | ||
}); | ||||
MinRK
|
r13063 | } | ||
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) { | ||
// Clears the navigation tree. | ||||
// | ||||
// Parameters | ||||
// remove_uploads: bool=False | ||||
// Should upload prompts also be removed from the tree. | ||||
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 () { | ||
Matthias BUSSONNIER
|
r7881 | var that = this; | ||
Brian E. Granger
|
r4488 | var settings = { | ||
processData : false, | ||||
cache : false, | ||||
type : "GET", | ||||
dataType : "json", | ||||
Matthias BUSSONNIER
|
r7881 | success : $.proxy(this.list_loaded, this), | ||
MinRK
|
r16445 | error : $.proxy( function(xhr, status, error){ | ||
Jonathan Frederic
|
r17198 | utils.log_ajax_error(xhr, status, error); | ||
Matthias BUSSONNIER
|
r7941 | that.list_loaded([], null, null, {msg:"Error connecting to server."}); | ||
},this) | ||||
Brian E. Granger
|
r4488 | }; | ||
Matthias BUSSONNIER
|
r7881 | |||
Jonathan Frederic
|
r17198 | var url = utils.url_join_encode( | ||
MinRK
|
r15238 | this.base_url, | ||
MinRK
|
r13063 | 'api', | ||
'notebooks', | ||||
MinRK
|
r15234 | this.notebook_path | ||
MinRK
|
r13063 | ); | ||
Brian E. Granger
|
r5106 | $.ajax(url, settings); | ||
Brian E. Granger
|
r4488 | }; | ||
Matthias BUSSONNIER
|
r7881 | NotebookList.prototype.list_loaded = function (data, status, xhr, param) { | ||
MinRK
|
r7976 | var message = 'Notebook list empty.'; | ||
if (param !== undefined && param.msg) { | ||||
MinRK
|
r13063 | message = param.msg; | ||
MinRK
|
r7976 | } | ||
Brian E. Granger
|
r15078 | var item = null; | ||
Brian E. Granger
|
r4488 | var len = data.length; | ||
Matthias BUSSONNIER
|
r6842 | this.clear_list(); | ||
MinRK
|
r13063 | if (len === 0) { | ||
MinRK
|
r15094 | item = this.new_notebook_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; | ||
Brian E. Granger
|
r15071 | var offset = 0; | ||
if (path !== '') { | ||||
Brian E. Granger
|
r15078 | item = this.new_notebook_item(0); | ||
Brian E. Granger
|
r15071 | this.add_dir(path, '..', item); | ||
offset = 1; | ||||
} | ||||
Brian E. Granger
|
r4488 | for (var i=0; i<len; i++) { | ||
Brian E. Granger
|
r15071 | if (data[i].type === 'directory') { | ||
var name = data[i].name; | ||||
Brian E. Granger
|
r15078 | item = this.new_notebook_item(i+offset); | ||
Brian E. Granger
|
r15071 | this.add_dir(path, name, item); | ||
MinRK
|
r11644 | } else { | ||
Brian E. Granger
|
r15071 | var name = data[i].name; | ||
Brian E. Granger
|
r15078 | item = this.new_notebook_item(i+offset); | ||
MinRK
|
r15094 | this.add_link(path, name, item); | ||
Jonathan Frederic
|
r17198 | name = utils.url_path_join(path, name); | ||
Brian E. Granger
|
r15071 | if(this.sessions[name] === undefined){ | ||
this.add_delete_button(item); | ||||
} else { | ||||
this.add_shutdown_button(item,this.sessions[name]); | ||||
} | ||||
MinRK
|
r5200 | } | ||
MinRK
|
r13063 | } | ||
Brian E. Granger
|
r4491 | }; | ||
Brian E. Granger
|
r4490 | |||
Brian E. Granger
|
r4491 | |||
NotebookList.prototype.new_notebook_item = function (index) { | ||||
Jonathan Frederic
|
r16913 | var item = $('<div/>').addClass("list_item").addClass("row"); | ||
MinRK
|
r10891 | // item.addClass('list_item ui-widget ui-widget-content ui-helper-clearfix'); | ||
// item.css('border-top-style','none'); | ||||
Jonathan Frederic
|
r16923 | item.append($("<div/>").addClass("col-md-12").append( | ||
Brian E. Granger
|
r15071 | $('<i/>').addClass('item_icon') | ||
).append( | ||||
MinRK
|
r10919 | $("<a/>").addClass("item_link").append( | ||
MinRK
|
r10891 | $("<span/>").addClass("item_name") | ||
) | ||||
MinRK
|
r10919 | ).append( | ||
$('<div/>').addClass("item_buttons btn-group pull-right") | ||||
)); | ||||
MinRK
|
r10891 | |||
Brian E. Granger
|
r4491 | if (index === -1) { | ||
Brian E. Granger
|
r4490 | this.element.append(item); | ||
Brian E. Granger
|
r4491 | } else { | ||
this.element.children().eq(index).after(item); | ||||
Brian E. Granger
|
r4488 | } | ||
Brian E. Granger
|
r4491 | return item; | ||
}; | ||||
Brian E. Granger
|
r15071 | NotebookList.prototype.add_dir = function (path, name, item) { | ||
item.data('name', name); | ||||
item.data('path', path); | ||||
item.find(".item_name").text(name); | ||||
Paul Ivanov
|
r16258 | item.find(".item_icon").addClass('folder_icon').addClass('icon-fixed-width'); | ||
Brian E. Granger
|
r15071 | item.find("a.item_link") | ||
.attr('href', | ||||
Jonathan Frederic
|
r17198 | utils.url_join_encode( | ||
MinRK
|
r15238 | this.base_url, | ||
Brian E. Granger
|
r15071 | "tree", | ||
path, | ||||
name | ||||
) | ||||
); | ||||
}; | ||||
Zachary Sailer
|
r12988 | NotebookList.prototype.add_link = function (path, nbname, item) { | ||
Brian E. Granger
|
r4491 | item.data('nbname', nbname); | ||
Zachary Sailer
|
r12988 | item.data('path', path); | ||
MinRK
|
r15094 | item.find(".item_name").text(nbname); | ||
Paul Ivanov
|
r16258 | item.find(".item_icon").addClass('notebook_icon').addClass('icon-fixed-width'); | ||
MinRK
|
r10919 | item.find("a.item_link") | ||
MinRK
|
r13063 | .attr('href', | ||
Jonathan Frederic
|
r17198 | utils.url_join_encode( | ||
MinRK
|
r15238 | this.base_url, | ||
MinRK
|
r13063 | "notebooks", | ||
MinRK
|
r13693 | path, | ||
MinRK
|
r15094 | nbname | ||
MinRK
|
r13063 | ) | ||
).attr('target','_blank'); | ||||
Brian E. Granger
|
r4491 | }; | ||
NotebookList.prototype.add_name_input = function (nbname, item) { | ||||
item.data('nbname', nbname); | ||||
Paul Ivanov
|
r16258 | item.find(".item_icon").addClass('notebook_icon').addClass('icon-fixed-width'); | ||
MinRK
|
r10919 | item.find(".item_name").empty().append( | ||
MinRK
|
r10896 | $('<input/>') | ||
.addClass("nbname_input") | ||||
Jonathan Frederic
|
r17198 | .attr('value', utils.splitext(nbname)[0]) | ||
MinRK
|
r10896 | .attr('size', '30') | ||
.attr('type', 'text') | ||||
Brian E. Granger
|
r4491 | ); | ||
}; | ||||
NotebookList.prototype.add_notebook_data = function (data, item) { | ||||
MinRK
|
r13072 | item.data('nbdata', data); | ||
Brian E. Granger
|
r4491 | }; | ||
Zachary Sailer
|
r12988 | NotebookList.prototype.add_shutdown_button = function (item, session) { | ||
Matthias BUSSONNIER
|
r6842 | var that = this; | ||
Jonathan Frederic
|
r16914 | var shutdown_button = $("<button/>").text("Shutdown").addClass("btn btn-xs btn-danger"). | ||
Matthias BUSSONNIER
|
r6842 | click(function (e) { | ||
var settings = { | ||||
processData : false, | ||||
cache : false, | ||||
type : "DELETE", | ||||
dataType : "json", | ||||
Zachary Sailer
|
r12996 | success : function () { | ||
Zachary Sailer
|
r12988 | that.load_sessions(); | ||
MinRK
|
r16445 | }, | ||
Jonathan Frederic
|
r17198 | error : utils.log_ajax_error, | ||
Matthias BUSSONNIER
|
r6842 | }; | ||
Jonathan Frederic
|
r17198 | var url = utils.url_join_encode( | ||
MinRK
|
r15238 | that.base_url, | ||
MinRK
|
r13063 | 'api/sessions', | ||
session | ||||
); | ||||
Matthias BUSSONNIER
|
r6842 | $.ajax(url, settings); | ||
MinRK
|
r10891 | return false; | ||
Matthias BUSSONNIER
|
r6842 | }); | ||
MinRK
|
r10891 | // var new_buttons = item.find('a'); // shutdown_button; | ||
Matthias BUSSONNIER
|
r14634 | item.find(".item_buttons").text("").append(shutdown_button); | ||
Matthias BUSSONNIER
|
r6842 | }; | ||
Brian E. Granger
|
r4491 | NotebookList.prototype.add_delete_button = function (item) { | ||
MinRK
|
r10891 | var new_buttons = $('<span/>').addClass("btn-group pull-right"); | ||
Matthias BUSSONNIER
|
r9787 | var notebooklist = this; | ||
Jonathan Frederic
|
r16913 | var delete_button = $("<button/>").text("Delete").addClass("btn btn-default btn-xs"). | ||
Brian E. Granger
|
r4491 | click(function (e) { | ||
// $(this) is the button that was clicked. | ||||
var that = $(this); | ||||
// We use the nbname and notebook_id from the parent notebook_item element's | ||||
// data because the outer scopes values change as we iterate through the loop. | ||||
Brian Granger
|
r6195 | var parent_item = that.parents('div.list_item'); | ||
Brian E. Granger
|
r4491 | var nbname = parent_item.data('nbname'); | ||
MinRK
|
r10921 | var message = 'Are you sure you want to permanently delete the notebook: ' + nbname + '?'; | ||
Jonathan Frederic
|
r17202 | dialog.modal({ | ||
MinRK
|
r10921 | title : "Delete notebook", | ||
body : message, | ||||
Brian E. Granger
|
r4491 | buttons : { | ||
MinRK
|
r10921 | Delete : { | ||
class: "btn-danger", | ||||
click: function() { | ||||
var settings = { | ||||
processData : false, | ||||
cache : false, | ||||
type : "DELETE", | ||||
dataType : "json", | ||||
success : function (data, status, xhr) { | ||||
parent_item.remove(); | ||||
MinRK
|
r16445 | }, | ||
Jonathan Frederic
|
r17198 | error : utils.log_ajax_error, | ||
MinRK
|
r10921 | }; | ||
Jonathan Frederic
|
r17198 | var url = utils.url_join_encode( | ||
MinRK
|
r15238 | notebooklist.base_url, | ||
MinRK
|
r13063 | 'api/notebooks', | ||
MinRK
|
r15234 | notebooklist.notebook_path, | ||
MinRK
|
r15094 | nbname | ||
MinRK
|
r13063 | ); | ||
MinRK
|
r10921 | $.ajax(url, settings); | ||
} | ||||
Brian E. Granger
|
r4491 | }, | ||
MinRK
|
r10921 | Cancel : {} | ||
Brian E. Granger
|
r4491 | } | ||
}); | ||||
MinRK
|
r10891 | return false; | ||
Brian E. Granger
|
r4491 | }); | ||
Matthias BUSSONNIER
|
r14634 | item.find(".item_buttons").text("").append(delete_button); | ||
Brian E. Granger
|
r4491 | }; | ||
NotebookList.prototype.add_upload_button = function (item) { | ||||
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
|
r13074 | var nbname = item.find('.item_name > input').val(); | ||
MinRK
|
r15094 | if (nbname.slice(nbname.length-6, nbname.length) != ".ipynb") { | ||
nbname = nbname + ".ipynb"; | ||||
} | ||||
MinRK
|
r15234 | var path = that.notebook_path; | ||
Brian E. Granger
|
r4491 | var nbdata = item.data('nbdata'); | ||
MinRK
|
r13072 | var content_type = 'application/json'; | ||
var model = { | ||||
content : JSON.parse(nbdata), | ||||
}; | ||||
Brian E. Granger
|
r4491 | var settings = { | ||
processData : false, | ||||
cache : false, | ||||
Matthias BUSSONNIER
|
r13322 | type : 'PUT', | ||
Brian E. Granger
|
r4493 | dataType : 'json', | ||
MinRK
|
r13072 | data : JSON.stringify(model), | ||
Brian E. Granger
|
r4493 | headers : {'Content-Type': content_type}, | ||
Brian E. Granger
|
r4491 | success : function (data, status, xhr) { | ||
Brian E. Granger
|
r14921 | that.add_link(path, nbname, item); | ||
Brian E. Granger
|
r4491 | that.add_delete_button(item); | ||
MinRK
|
r13072 | }, | ||
Jonathan Frederic
|
r17198 | error : utils.log_ajax_error, | ||
Brian E. Granger
|
r4491 | }; | ||
Jonathan Frederic
|
r17198 | var url = utils.url_join_encode( | ||
MinRK
|
r15238 | that.base_url, | ||
MinRK
|
r13072 | 'api/notebooks', | ||
MinRK
|
r15234 | that.notebook_path, | ||
MinRK
|
r15094 | nbname | ||
MinRK
|
r13063 | ); | ||
Brian E. Granger
|
r5106 | $.ajax(url, settings); | ||
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) { | ||
console.log('cancel click'); | ||||
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 | }; | ||
Zachary Sailer
|
r13017 | NotebookList.prototype.new_notebook = function(){ | ||
MinRK
|
r15234 | var path = this.notebook_path; | ||
MinRK
|
r15238 | var base_url = this.base_url; | ||
Zachary Sailer
|
r13017 | var settings = { | ||
processData : false, | ||||
cache : false, | ||||
type : "POST", | ||||
dataType : "json", | ||||
MinRK
|
r13075 | async : false, | ||
success : function (data, status, xhr) { | ||||
MinRK
|
r13063 | var notebook_name = data.name; | ||
window.open( | ||||
Jonathan Frederic
|
r17198 | utils.url_join_encode( | ||
MinRK
|
r15238 | base_url, | ||
MinRK
|
r13063 | 'notebooks', | ||
MinRK
|
r13069 | path, | ||
MinRK
|
r13063 | notebook_name), | ||
'_blank' | ||||
); | ||||
MinRK
|
r16445 | }, | ||
MinRK
|
r16446 | error : $.proxy(this.new_notebook_failed, this), | ||
Zachary Sailer
|
r13017 | }; | ||
Jonathan Frederic
|
r17198 | var url = utils.url_join_encode( | ||
MinRK
|
r15238 | base_url, | ||
MinRK
|
r13063 | 'api/notebooks', | ||
path | ||||
); | ||||
$.ajax(url, settings); | ||||
Zachary Sailer
|
r13017 | }; | ||
MinRK
|
r16446 | |||
NotebookList.prototype.new_notebook_failed = function (xhr, status, error) { | ||||
Jonathan Frederic
|
r17198 | utils.log_ajax_error(xhr, status, error); | ||
MinRK
|
r16446 | var msg; | ||
if (xhr.responseJSON && xhr.responseJSON.message) { | ||||
msg = xhr.responseJSON.message; | ||||
} else { | ||||
msg = xhr.statusText; | ||||
} | ||||
Jonathan Frederic
|
r17202 | dialog.modal({ | ||
MinRK
|
r16446 | title : 'Creating Notebook Failed', | ||
body : "The error was: " + msg, | ||||
buttons : {'OK' : {'class' : 'btn-primary'}} | ||||
}); | ||||
Jonathan Frederic
|
r17189 | }; | ||
MinRK
|
r16446 | |||
Jonathan Frederic
|
r17189 | // Backwards compatability. | ||
Brian E. Granger
|
r4488 | IPython.NotebookList = NotebookList; | ||
Jonathan Frederic
|
r17201 | return {'NotebookList': NotebookList}; | ||
Jonathan Frederic
|
r17189 | }); | ||