savewidget.js
254 lines
| 8.8 KiB
| application/javascript
|
JavascriptLexer
Jonathan Frederic
|
r17198 | // Copyright (c) IPython Development Team. | ||
// Distributed under the terms of the Modified BSD License. | ||||
define([ | ||||
'base/js/namespace', | ||||
Jonathan Frederic
|
r17200 | 'jquery', | ||
Jonathan Frederic
|
r17198 | 'base/js/utils', | ||
'base/js/dialog', | ||||
'base/js/keyboard', | ||||
Matthias BUSSONNIER
|
r17474 | 'moment', | ||
], function(IPython, $, utils, dialog, keyboard, moment) { | ||||
Matthias BUSSONNIER
|
r12103 | "use strict"; | ||
Brian E. Granger
|
r4369 | |||
Jonathan Frederic
|
r17213 | var SaveWidget = function (selector, options) { | ||
Matthias BUSSONNIER
|
r17442 | // TODO: Remove circular ref. | ||
Jonathan Frederic
|
r17198 | this.notebook = undefined; | ||
Brian E. Granger
|
r4372 | this.selector = selector; | ||
Jonathan Frederic
|
r17213 | this.events = options.events; | ||
Matthias BUSSONNIER
|
r17474 | this._checkpoint_date = undefined; | ||
Jonathan Frederic
|
r17213 | this.keyboard_manager = options.keyboard_manager; | ||
Brian E. Granger
|
r4372 | if (this.selector !== undefined) { | ||
this.element = $(selector); | ||||
Brian E. Granger
|
r4369 | this.bind_events(); | ||
} | ||||
}; | ||||
Brian E. Granger
|
r4372 | |||
Brian E. Granger
|
r4369 | SaveWidget.prototype.bind_events = function () { | ||
var that = this; | ||||
Brian Granger
|
r5859 | this.element.find('span#notebook_name').click(function () { | ||
Min RK
|
r18964 | that.rename_notebook({notebook: that.notebook}); | ||
Brian E. Granger
|
r4372 | }); | ||
Jonathan Frederic
|
r17198 | this.events.on('notebook_loaded.Notebook', function () { | ||
Brian Granger
|
r6047 | that.update_notebook_name(); | ||
that.update_document_title(); | ||||
}); | ||||
Jonathan Frederic
|
r17198 | this.events.on('notebook_saved.Notebook', function () { | ||
Brian Granger
|
r6047 | that.update_notebook_name(); | ||
that.update_document_title(); | ||||
}); | ||||
Jonathan Frederic
|
r17198 | this.events.on('notebook_renamed.Notebook', function () { | ||
Zachary Sailer
|
r13012 | that.update_notebook_name(); | ||
that.update_document_title(); | ||||
Zachary Sailer
|
r13010 | that.update_address_bar(); | ||
Zachary Sailer
|
r12997 | }); | ||
Jonathan Frederic
|
r17198 | this.events.on('notebook_save_failed.Notebook', function () { | ||
MinRK
|
r10829 | that.set_save_status('Autosave Failed!'); | ||
Brian Granger
|
r6047 | }); | ||
Min RK
|
r19005 | this.events.on('notebook_read_only.Notebook', function () { | ||
that.set_save_status('(read only)'); | ||||
// disable future set_save_status | ||||
that.set_save_status = function () {}; | ||||
}); | ||||
Jonathan Frederic
|
r17198 | this.events.on('checkpoints_listed.Notebook', function (event, data) { | ||
Matthias BUSSONNIER
|
r17474 | that._set_last_checkpoint(data[0]); | ||
MinRK
|
r10516 | }); | ||
Kevin Burke
|
r12368 | |||
Jonathan Frederic
|
r17198 | this.events.on('checkpoint_created.Notebook', function (event, data) { | ||
Matthias BUSSONNIER
|
r17474 | that._set_last_checkpoint(data); | ||
MinRK
|
r10516 | }); | ||
Jonathan Frederic
|
r17198 | this.events.on('set_dirty.Notebook', function (event, data) { | ||
MinRK
|
r10829 | that.set_autosaved(data.value); | ||
}); | ||||
Felix Werner
|
r5006 | }; | ||
Jonathan Frederic
|
r17213 | SaveWidget.prototype.rename_notebook = function (options) { | ||
options = options || {}; | ||||
Brian Granger
|
r5859 | var that = this; | ||
Jonathan Frederic
|
r17203 | var dialog_body = $('<div/>').append( | ||
MinRK
|
r10956 | $("<p/>").addClass("rename-message") | ||
Matthias BUSSONNIER
|
r14634 | .text('Enter a new notebook name:') | ||
MinRK
|
r10956 | ).append( | ||
$("<br/>") | ||||
).append( | ||||
Jonathan Frederic
|
r16957 | $('<input/>').attr('type','text').attr('size','25').addClass('form-control') | ||
Min RK
|
r18964 | .val(options.notebook.get_notebook_name()) | ||
Brian Granger
|
r5859 | ); | ||
Min RK
|
r18964 | var d = dialog.modal({ | ||
Brian Granger
|
r5859 | title: "Rename Notebook", | ||
Jonathan Frederic
|
r17203 | body: dialog_body, | ||
Jonathan Frederic
|
r17213 | notebook: options.notebook, | ||
keyboard_manager: this.keyboard_manager, | ||||
Brian Granger
|
r5859 | buttons : { | ||
MinRK
|
r10895 | "OK": { | ||
class: "btn-primary", | ||||
click: function () { | ||||
Min RK
|
r18964 | var new_name = d.find('input').val(); | ||
if (!options.notebook.test_notebook_name(new_name)) { | ||||
d.find('.rename-message').text( | ||||
"Invalid notebook name. Notebook names must "+ | ||||
"have 1 or more characters and can contain any characters " + | ||||
"except :/\\. Please enter a new notebook name:" | ||||
); | ||||
return false; | ||||
} else { | ||||
d.find('.rename-message').text("Renaming..."); | ||||
d.find('input[type="text"]').prop('disabled', true); | ||||
that.notebook.rename(new_name).then( | ||||
function () { | ||||
d.modal('hide'); | ||||
}, function (error) { | ||||
d.find('.rename-message').text(error.message || 'Unknown error'); | ||||
d.find('input[type="text"]').prop('disabled', false).focus().select(); | ||||
} | ||||
); | ||||
return false; | ||||
} | ||||
Brian Granger
|
r5859 | } | ||
Min RK
|
r18964 | }, | ||
David Neto
|
r18926 | "Cancel": {} | ||
Brian Granger
|
r5859 | }, | ||
Min RK
|
r18964 | open : function () { | ||
Brian Granger
|
r7246 | // Upon ENTER, click the OK button. | ||
Min RK
|
r18964 | d.find('input[type="text"]').keydown(function (event) { | ||
Jonathan Frederic
|
r17213 | if (event.which === keyboard.keycodes.enter) { | ||
Min RK
|
r18964 | d.find('.btn-primary').first().click(); | ||
MinRK
|
r12019 | return false; | ||
Brian Granger
|
r7246 | } | ||
}); | ||||
Min RK
|
r18964 | d.find('input[type="text"]').focus().select(); | ||
Brian Granger
|
r5859 | } | ||
}); | ||||
Jonathan Frederic
|
r17198 | }; | ||
Brian Granger
|
r5859 | |||
Brian E. Granger
|
r4645 | |||
Brian Granger
|
r6047 | SaveWidget.prototype.update_notebook_name = function () { | ||
Jonathan Frederic
|
r17198 | var nbname = this.notebook.get_notebook_name(); | ||
Matthias BUSSONNIER
|
r14634 | this.element.find('span#notebook_name').text(nbname); | ||
Stefan van der Walt
|
r5479 | }; | ||
Brian E. Granger
|
r4372 | |||
Brian E. Granger
|
r4369 | |||
Brian Granger
|
r6047 | SaveWidget.prototype.update_document_title = function () { | ||
Jonathan Frederic
|
r17198 | var nbname = this.notebook.get_notebook_name(); | ||
Brian E. Granger
|
r5104 | document.title = nbname; | ||
Brian E. Granger
|
r4549 | }; | ||
Matthias BUSSONNIER
|
r17474 | |||
Zachary Sailer
|
r13010 | SaveWidget.prototype.update_address_bar = function(){ | ||
Jonathan Frederic
|
r17198 | var base_url = this.notebook.base_url; | ||
var path = this.notebook.notebook_path; | ||||
Min RK
|
r18752 | var state = {path : path}; | ||
MinRK
|
r13693 | window.history.replaceState(state, "", utils.url_join_encode( | ||
MinRK
|
r15483 | base_url, | ||
"notebooks", | ||||
Min RK
|
r18752 | path) | ||
MinRK
|
r13103 | ); | ||
MinRK
|
r15483 | }; | ||
Brian E. Granger
|
r4484 | |||
Brian Granger
|
r6047 | SaveWidget.prototype.set_save_status = function (msg) { | ||
Matthias BUSSONNIER
|
r14634 | this.element.find('span#autosave_status').text(msg); | ||
Jonathan Frederic
|
r17198 | }; | ||
MinRK
|
r10516 | |||
Matthias BUSSONNIER
|
r17474 | SaveWidget.prototype._set_checkpoint_status = function (human_date, iso_date) { | ||
MinRK
|
r17715 | var el = this.element.find('span#checkpoint_status'); | ||
Matthias BUSSONNIER
|
r17474 | if(human_date){ | ||
el.text("Last Checkpoint: "+human_date).attr('title',iso_date); | ||||
} else { | ||||
MinRK
|
r17715 | el.text('').attr('title', 'no-checkpoint'); | ||
Matthias BUSSONNIER
|
r17474 | } | ||
Jonathan Frederic
|
r17198 | }; | ||
Brian E. Granger
|
r4484 | |||
Matthias BUSSONNIER
|
r17474 | // compute (roughly) the remaining time in millisecond until the next | ||
// moment.js relative time update of the string, which by default | ||||
// happend at | ||||
// (a few seconds ago) | ||||
// - 45sec, | ||||
// (a minute ago) | ||||
// - 90sec, | ||||
// ( x minutes ago) | ||||
// - then every minutes until | ||||
// - 45 min, | ||||
// (an hour ago) | ||||
// - 1h45, | ||||
// (x hours ago ) | ||||
// - then every hours | ||||
// - 22 hours ago | ||||
var _next_timeago_update = function(deltatime_ms){ | ||||
var s = 1000; | ||||
var m = 60*s; | ||||
var h = 60*m; | ||||
var mtt = moment.relativeTimeThreshold; | ||||
if(deltatime_ms < mtt.s*s){ | ||||
return mtt.s*s-deltatime_ms; | ||||
} else if (deltatime_ms < (mtt.s*s+m)) { | ||||
return (mtt.s*s+m)-deltatime_ms; | ||||
} else if (deltatime_ms < mtt.m*m){ | ||||
return m; | ||||
} else if (deltatime_ms < (mtt.m*m+h)){ | ||||
return (mtt.m*m+h)-deltatime_ms; | ||||
} else { | ||||
return h; | ||||
} | ||||
MinRK
|
r17715 | }; | ||
Matthias BUSSONNIER
|
r17474 | |||
SaveWidget.prototype._regularly_update_checkpoint_date = function(){ | ||||
if (!this._checkpoint_date) { | ||||
MinRK
|
r17715 | this._set_checkpoint_status(null); | ||
Matthias BUSSONNIER
|
r17474 | console.log('no checkpoint done'); | ||
MinRK
|
r10547 | return; | ||
MinRK
|
r10532 | } | ||
Matthias BUSSONNIER
|
r17474 | var chkd = moment(this._checkpoint_date); | ||
var longdate = chkd.format('llll'); | ||||
var that = this; | ||||
var recall = function(t){ | ||||
// recall slightly later (1s) as long timeout in js might be imprecise, | ||||
// and you want to be call **after** the change of formatting should happend. | ||||
MinRK
|
r17715 | return setTimeout( | ||
$.proxy(that._regularly_update_checkpoint_date, that), | ||||
t + 1000 | ||||
); | ||||
Min RK
|
r18752 | }; | ||
Matthias BUSSONNIER
|
r17474 | var tdelta = Math.ceil(new Date()-this._checkpoint_date); | ||
// update regularly for the first 6hours and show | ||||
// <x time> ago | ||||
if(tdelta < tdelta < 6*3600*1000){ | ||||
recall(_next_timeago_update(tdelta)); | ||||
MinRK
|
r17715 | this._set_checkpoint_status(chkd.fromNow(), longdate); | ||
Matthias BUSSONNIER
|
r17474 | // otherwise update every hour and show | ||
// <Today | yesterday|...> at hh,mm,ss | ||||
} else { | ||||
MinRK
|
r17715 | recall(1*3600*1000); | ||
this._set_checkpoint_status(chkd.calendar(), longdate); | ||||
Matthias BUSSONNIER
|
r17474 | } | ||
MinRK
|
r17715 | }; | ||
Matthias BUSSONNIER
|
r17474 | |||
SaveWidget.prototype._set_last_checkpoint = function (checkpoint) { | ||||
MinRK
|
r17715 | if (checkpoint) { | ||
this._checkpoint_date = new Date(checkpoint.last_modified); | ||||
} else { | ||||
this._checkpoint_date = null; | ||||
} | ||||
Matthias BUSSONNIER
|
r17474 | this._regularly_update_checkpoint_date(); | ||
Jonathan Frederic
|
r17198 | }; | ||
Brian E. Granger
|
r4484 | |||
MinRK
|
r10829 | SaveWidget.prototype.set_autosaved = function (dirty) { | ||
if (dirty) { | ||||
this.set_save_status("(unsaved changes)"); | ||||
} else { | ||||
this.set_save_status("(autosaved)"); | ||||
} | ||||
Brian E. Granger
|
r4630 | }; | ||
Matthias BUSSONNIER
|
r17443 | // Backwards compatibility. | ||
Brian E. Granger
|
r4369 | IPython.SaveWidget = SaveWidget; | ||
Jonathan Frederic
|
r17201 | return {'SaveWidget': SaveWidget}; | ||
Brian E. Granger
|
r4369 | |||
Jonathan Frederic
|
r17198 | }); | ||