| $.Class ("ChronoShare", { }, |
| { |
| init: function (username, foldername) { |
| $("#folder-name").text (foldername); |
| $("#user-name").text (username); |
| |
| this.username = new Name (username); |
| this.files = new Name ("/localhost").add (this.username).add ("chronoshare").add (foldername).add ("info").add ("files").add ("folder"); |
| |
| this.actions = new Name ("/localhost").add (this.username).add ("chronoshare").add (foldername).add ("info").add ("actions"); |
| |
| this.restore = new Name ("/localhost").add (this.username).add ("chronoshare").add (foldername).add ("cmd").add ("restore").add ("file"); |
| |
| this.ndn = new NDN ({host:"127.0.0.1"}); |
| this.ndn.verify = false; //disable content verification, works WAAAAY faster |
| }, |
| |
| run: function () { |
| console.log ("RUN page: " + PAGE); |
| $("#loader").fadeIn (500); |
| $("#error").addClass ("hidden"); |
| |
| cmd = {}; |
| if (PAGE == "fileList") { |
| cmd = this.info_files (PARAMS.item); |
| } |
| else if (PAGE == "folderHistory") { |
| cmd = this.info_actions ("folder", PARAMS.item); |
| } |
| else if (PAGE == "fileHistory") { |
| cmd = this.info_actions ("file", PARAMS.item); |
| } |
| |
| if (cmd.request && cmd.callback) { |
| console.log (cmd.request.to_uri ()); |
| this.ndn.expressInterest (cmd.request, cmd.callback); |
| } |
| else { |
| $("#loader").fadeOut (500); // ("hidden"); |
| $("#content").empty (); |
| if (cmd.error) { |
| $("#error").html (cmd.error); |
| } |
| else { |
| $("#error").html ("Unknown error with " + PAGE); |
| } |
| $("#error").removeClass ("hidden"); |
| } |
| }, |
| |
| info_files: function(folder) { |
| request = new Name ().add (this.files)./*add (folder_in_question).*/addSegment (PARAMS.offset?PARAMS.offset:0); |
| return { request:request, callback: new FilesClosure (this) }; |
| }, |
| |
| info_actions: function (type/*"file" or "folder"*/, fileOrFolder /*file or folder name*/) { |
| if (type=="file" && !fileOrFolder) { |
| return { error: "info_actions: fileOrFolder parameter is missing" }; |
| } |
| |
| request = new Name ().add (this.actions).add (type); |
| if (fileOrFolder) { |
| request.add (fileOrFolder); |
| } |
| request.addSegment (PARAMS.offset?PARAMS.offset:0); |
| return { request: request, callback: new HistoryClosure (this) }; |
| }, |
| |
| cmd_restore_file: function (filename, version, hash, callback/*function (bool <- data received, status <- returned status)*/) { |
| request = new Name ().add (this.restore) |
| .add (filename) |
| .addSegment (version) |
| .add (hash); |
| console.log (request.to_uri ()); |
| this.ndn.expressInterest (request, new CmdRestoreFileClosure (this, callback)); |
| }, |
| |
| get_file: function (modifiedBy, hash, segments, callback/*function (bool <- data received, data <- returned data)*/) { |
| baseName = new Name (modifiedBy) |
| .add ("chronoshare").add ("file") |
| .add (hash); |
| |
| new FileGetter (this.ndn, baseName, segments, callback) |
| .start (); |
| } |
| }); |
| |
| $.Class ("CmdRestoreFileClosure", {}, { |
| init: function (chronoshare, callback) { |
| this.chronoshare = chronoshare; |
| this.callback = callback; |
| }, |
| upcall: function(kind, upcallInfo) { |
| if (kind == Closure.UPCALL_CONTENT || kind == Closure.UPCALL_CONTENT_UNVERIFIED) { //disable content verification |
| convertedData = DataUtils.toString (upcallInfo.contentObject.content); |
| this.callback (true, convertedData); |
| } |
| else if (kind == Closure.UPCALL_INTEREST_TIMED_OUT) { |
| this.callback (false, "Interest timed out"); |
| } |
| else { |
| this.callback (false, "Unknown error happened"); |
| } |
| } |
| }); |
| |
| $.Class ("FileGetter", {}, { |
| init: function (ndn, baseName, segments, callback) { |
| this.ndn = ndn; |
| this.callback = callback; |
| this.baseName = baseName; |
| this.segments = segments; |
| this.lastSegmentRequested = -1; |
| |
| this.data = ""; |
| }, |
| |
| start: function () { |
| this.lastSegmentRequested ++; |
| |
| request = new Name () |
| .add (this.baseName) |
| .addSegment (this.lastSegmentRequested); |
| |
| console.log (request.to_uri ()); |
| this.ndn.expressInterest (request, this); |
| }, |
| |
| upcall: function(kind, upcallInfo) { |
| if (kind == Closure.UPCALL_CONTENT || kind == Closure.UPCALL_CONTENT_UNVERIFIED) { //disable content verification |
| convertedData = DataUtils.toString (upcallInfo.contentObject.content); |
| |
| this.data += convertedData; |
| |
| if (this.lastSegmentRequested+1 == this.segments) { |
| this.callback (true, this.data); |
| } |
| else { |
| this.start (); |
| } |
| } |
| else { |
| this.callback (false, "Interest timed out"); |
| } |
| } |
| }); |
| |
| $.Class ("RestPipelineClosure", {}, { |
| init: function (collectionName, moreName) { |
| this.collectionName = collectionName; |
| this.moreName = moreName; |
| $("#json").empty (); |
| |
| this.collection = []; |
| this.counter = 0; |
| }, |
| |
| upcall: function(kind, upcallInfo) { |
| if (kind == Closure.UPCALL_CONTENT || kind == Closure.UPCALL_CONTENT_UNVERIFIED) { //disable content verification |
| |
| convertedData = DataUtils.toString (upcallInfo.contentObject.content); |
| if (PARAMS.debug) { |
| $("#json").append ($(document.createTextNode(convertedData))); |
| $("#json").removeClass ("hidden"); |
| } |
| data = JSON.parse (convertedData); |
| |
| this.collection = this.collection.concat (data[this.collectionName]); |
| if (data[this.moreName] !== undefined) { |
| nextSegment = upcallInfo.interest.name.cut (1).addSegment (data[this.moreName]); |
| this.counter ++; |
| |
| if (this.counter < 5) { |
| console.log ("MORE: " +nextSegment.to_uri ()); |
| CHRONOSHARE.ndn.expressInterest (nextSegment, this); |
| } |
| else { |
| $("#loader").fadeOut (500); // ("hidden"); |
| this.onData (this.collection, data[this.moreName]); |
| } |
| } |
| else { |
| $("#loader").fadeOut (500); // ("hidden"); |
| this.onData (this.collection, undefined); |
| } |
| } |
| else if (kind == Closure.UPCALL_INTEREST_TIMED_OUT) { |
| $("#loader").fadeOut (500); // ("hidden"); |
| this.onTimeout (upcallInfo.interest); |
| } |
| else { |
| $("#loader").fadeOut (500); // ("hidden"); |
| this.onUnknownError (upcallInfo.interest); |
| } |
| |
| return Closure.RESULT_OK; // make sure we never re-express the interest |
| }, |
| |
| onData: function(data, more) { |
| }, |
| |
| onTimeout: function () { |
| $("#error").html ("Interest timed out"); |
| $("#error").removeClass ("hidden"); |
| }, |
| |
| onUnknownError: function () { |
| $("#error").html ("Unknown error happened"); |
| $("#error").removeClass ("hidden"); |
| } |
| }); |
| |
| // $.Class ("FilesClosure", {}, { |
| RestPipelineClosure ("FilesClosure", {}, { |
| init: function (chronoshare) { |
| this._super("files", "more"); |
| this.chronoshare = chronoshare; |
| }, |
| |
| onData: function(data, more) { |
| tbody = $("<tbody />", { "id": "file-list-files" }); |
| |
| /// @todo Eventually set title for other pages |
| $("title").text ("ChronoShare - List of files" + (PARAMS.item?" - "+PARAMS.item:"")); |
| |
| // error handling? |
| newcontent = $("<div />", { "id": "content" }).append ( |
| $("<h2 />").append ($(document.createTextNode("List of files ")), $("<green />").text (PARAMS.item)), |
| $("<table />", { "class": "item-list" }) |
| .append ($("<thead />") |
| .append ($("<tr />") |
| .append ($("<th />", { "class": "filename border-left", "scope": "col" }).text ("Filename")) |
| .append ($("<th />", { "class": "version", "scope": "col" }).text ("Version")) |
| .append ($("<th />", { "class": "size", "scope": "col" }).text ("Size")) |
| .append ($("<th />", { "class": "modified", "scope": "col" }).text ("Modified")) |
| .append ($("<th />", { "class": "modified-by border-right", "scope": "col" }).text ("Modified By")))) |
| .append (tbody) |
| .append ($("<tfoot />") |
| .append ($("<tr />") |
| .append ($("<td />", { "colspan": "5", "class": "border-right border-left" }))))); |
| newcontent.hide (); |
| |
| for (var i = 0; i < data.length; i++) { |
| file = data[i]; |
| |
| row = $("<tr />", { "class": "with-context-menu" } ); |
| if (i%2) { row.addClass ("odd"); } |
| |
| row.bind('mouseenter mouseleave', function() { |
| $(this).toggleClass('highlighted'); |
| }); |
| |
| row.attr ("filename", file.filename); //encodeURIComponent(encodeURIComponent(file.filename))); |
| row.bind('click', function (e) { openHistoryForItem ($(this).attr ("filename")) }); |
| |
| row.append ($("<td />", { "class": "filename border-left" }) |
| .text (file.filename) |
| .prepend ($("<img />", { "src": imgFullPath(fileExtension(file.filename)) }))); |
| row.append ($("<td />", { "class": "version" }).text (file.version)); |
| row.append ($("<td />", { "class": "size" }).text (SegNumToFileSize (file.segNum))); |
| row.append ($("<td />", { "class": "modified" }).text (new Date (file.timestamp+"+00:00"))); // convert from UTC |
| row.append ($("<td />", { "class": "modified-by border-right"}) |
| .append ($("<userName />").text (file.owner.userName)) |
| .append ($("<seqNo> /").text (file.owner.seqNo))); |
| |
| tbody = tbody.append (row); |
| } |
| |
| displayContent (newcontent, more, this.base_url ()); |
| |
| $.contextMenu( 'destroy', ".with-context-menu" ); // cleanup |
| $.contextMenu({ |
| selector: ".with-context-menu", |
| items: { |
| "info": {name: "x", type: "html", html: "<b>File operations</b>"}, |
| "sep1": "---------", |
| history: {name: "View file history", |
| icon: "quit", // need a better icon |
| callback: function(key, opt) { |
| openHistoryForItem (opt.$trigger.attr ("filename")); |
| }}, |
| } |
| }); |
| }, |
| |
| base_url: function () { |
| url = "#fileList"+ |
| "&user="+encodeURIComponent (encodeURIComponent (PARAMS.user)) + |
| "&folder="+encodeURIComponent (encodeURIComponent (PARAMS.folder)); |
| if (PARAMS.item !== undefined) { |
| url += "&item="+encodeURIComponent (encodeURIComponent (PARAMS.item)); |
| } |
| return url; |
| } |
| }); |
| |
| |
| RestPipelineClosure ("HistoryClosure", {}, { |
| init: function (chronoshare) { |
| this._super("actions", "more"); |
| this.chronoshare = chronoshare; |
| }, |
| |
| previewFile: function (file) { |
| if (fileExtension(file.attr ("filename")) == "txt") { |
| CHRONOSHARE.get_file (file.attr ("file_modified_by"), |
| DataUtils.toNumbers (file.attr ("file_hash")), |
| file.attr ("file_seg_num"), |
| function (status, data) { |
| $("<div />", { "title": "Preview of " + file.attr ("filename") + " version " + file.attr ("file_version") }) |
| .append ($("<pre />").text (data)) |
| .dialog ({ |
| resizable: true, |
| width: $(window).width() * 0.8, |
| maxHeight: $(window).height() * 0.8, |
| show: "blind", |
| hide: "fold", |
| modal: true, |
| }); |
| }); |
| } |
| else { |
| custom_alert ("Preview is not support for this type of file"); |
| } |
| }, |
| |
| onData: function(data, more) { |
| tbody = $("<tbody />", { "id": "history-list-actions" }); |
| |
| /// @todo Eventually set title for other pages |
| $("title").text ("ChronoShare - Recent actions" + (PARAMS.item?" - "+PARAMS.item:"")); |
| |
| newcontent = $("<div />", { "id": "content" }).append ( |
| $("<h2 />").append ($(document.createTextNode("Recent actions ")), $("<green />").text (PARAMS.item)), |
| $("<table />", { "class": "item-list" }) |
| .append ($("<thead />") |
| .append ($("<tr />") |
| .append ($("<th />", { "class": "filename border-left", "scope": "col" }).text ("Filename")) |
| .append ($("<th />", { "class": "version", "scope": "col" }).text ("Version")) |
| .append ($("<th />", { "class": "size", "scope": "col" }).text ("Size")) |
| .append ($("<th />", { "class": "modified", "scope": "col" }).text ("Modified")) |
| .append ($("<th />", { "class": "modified-by border-right", "scope": "col" }).text ("Modified By")))) |
| .append (tbody) |
| .append ($("<tfoot />") |
| .append ($("<tr />") |
| .append ($("<td />", { "colspan": "5", "class": "border-right border-left" }))))); |
| |
| for (var i = 0; i < data.length; i++) { |
| action = data[i]; |
| |
| row = $("<tr />"); |
| if (i%2) { row.addClass ("odd"); } |
| if (action.action=="DELETE") { |
| row.addClass ("delete"); |
| } |
| else { |
| row.addClass ("with-context-menu"); |
| row.attr ("file_version", action.version); |
| row.attr ("file_hash", action.update.hash); |
| row.attr ("file_seg_num", action.update.segNum); |
| row.attr ("file_modified_by", action.id.userName); |
| } |
| |
| row.attr ("filename", action.filename); |
| |
| self = this; |
| if (PARAMS.item != action.filename) { |
| row.bind('click', function (e) { openHistoryForItem ($(this).attr ("filename")) }); |
| } |
| else { |
| row.bind('click', function (e) { |
| self.previewFile ($(this)); |
| }); |
| } |
| |
| row.bind('mouseenter mouseleave', function() { |
| $(this).toggleClass('highlighted'); |
| }); |
| |
| row.append ($("<td />", { "class": "filename border-left" }) |
| .text (action.filename) |
| .prepend ($("<img />", { "src": imgFullPath(fileExtension(action.filename)) }))); |
| row.append ($("<td />", { "class": "version" }).text (action.version)); |
| row.append ($("<td />", { "class": "size" }).text (action.update?SegNumToFileSize (action.update.segNum):"")); |
| row.append ($("<td />", { "class": "timestamp" }).text (new Date (action.timestamp+"+00:00"))); // conversion from UTC timezone (we store action time in UTC) |
| row.append ($("<td />", { "class": "modified-by border-right" }) |
| .append ($("<userName />").text (action.id.userName)) |
| .append ($("<seqNo> /").text (action.id.seqNo))); |
| |
| tbody = tbody.append (row); |
| } |
| |
| displayContent (newcontent, more, this.base_url (PAGE)); |
| |
| self = this; |
| $.contextMenu( 'destroy', ".with-context-menu" ); // cleanup |
| $.contextMenu({ |
| selector: ".with-context-menu", |
| items: { |
| "sep1": "---------", |
| preview: {name: "Preview revision", |
| icon: "edit", // ned a better icon |
| callback: function(key, opt) { |
| self.previewFile (opt.$trigger); |
| }}, |
| "sep3": "---------", |
| restore: {name: "Restore this revision", |
| icon: "cut", // need a better icon |
| callback: function(key, opt) { |
| filename = opt.$trigger.attr ("filename"); |
| version = opt.$trigger.attr ("file_version"); |
| hash = DataUtils.toNumbers (opt.$trigger.attr ("file_hash")); |
| console.log (hash); |
| modified_by = opt.$trigger.attr ("file_modified_by"); |
| |
| $("<div />", { "title": "Restore version " + version + "?" }) |
| .append ($("<p />") |
| .append ($("<span />", { "class": "ui-icon ui-icon-alert", |
| "style": "float: left; margin: 0 7px 50px 0;" }), |
| $(document.createTextNode ("Are you sure you want restore version ")), |
| $("<green/>").text (version), |
| $(document.createTextNode (" by ")), |
| $("<green/>").text (modified_by))) |
| .dialog ({ |
| resizable: true, |
| height:200, |
| width:300, |
| modal: true, |
| show: "blind", |
| hide: "fold", |
| buttons: { |
| "Restore": function() { |
| self = $(this); |
| CHRONOSHARE.cmd_restore_file (filename, version, hash, function(didGetData, response) { |
| if (!didGetData || response != "OK") { |
| custom_alert (response); |
| } |
| console.log (response); |
| self.dialog ("close"); |
| |
| $.timer (function() {CHRONOSHARE.run ();}).once (1000); |
| }); |
| }, |
| Cancel: function() { |
| $(this).dialog ("close"); |
| } |
| } |
| }); |
| // openHistoryForItem (opt.$trigger.attr ("filename")); |
| }}, |
| "sep2": "---------", |
| } |
| }); |
| }, |
| |
| base_no_item_url: function (page) { |
| url = "#"+page+ |
| "&user="+encodeURIComponent (encodeURIComponent (PARAMS.user)) + |
| "&folder="+encodeURIComponent (encodeURIComponent (PARAMS.folder)); |
| return url; |
| }, |
| |
| base_url: function (page) { |
| url = "#"+page+ |
| "&user="+encodeURIComponent (encodeURIComponent (PARAMS.user)) + |
| "&folder="+encodeURIComponent (encodeURIComponent (PARAMS.folder)); |
| if (PARAMS.item !== undefined) { |
| url += "&item="+encodeURIComponent (encodeURIComponent (PARAMS.item)); |
| } |
| return url; |
| } |
| }); |
| |
| |
| |
| |