Repaired sync down

This commit is contained in:
benweet 2013-04-03 23:52:29 +01:00
parent 6fae74a18e
commit ec9d93e64c
7 changed files with 187 additions and 105 deletions

View File

@ -10,3 +10,4 @@ RequireJS - http://requirejs.org/
PageDown - https://code.google.com/p/pagedown/
UI Layout - http://layout.jquery-dev.net/
jGrowl - https://github.com/stanlemon/jGrowl/
GLYPHICONS - http://glyphicons.com/

View File

@ -5,6 +5,10 @@ body {
background-color: #f5f5f5;
}
.working {
cursor: progress;
}
.btn {
-webkit-user-select: none;
-moz-user-select: none;
@ -25,12 +29,23 @@ div, span, a, ul, li, textarea, input, button {
text-shadow: none !important;
}
.btn, .navbar-inner, .ui-layout-east, .ui-layout-south, textarea, input, .input-append .add-on {
.btn, .navbar-inner, .ui-layout-east, .ui-layout-south, textarea, input, .add-on {
border: none !important;
}
.border, .dropdown-menu {
border: 1px solid #ddd !important;
.dropdown-menu {
border: 1px solid #e2e2e2 !important;
}
.input-prepend input,
.input-prepend .btn,
.input-prepend .add-on
{
border: 1px solid #ebebeb !important;
}
input.error {
border-color: #ee5f5b !important;
}
.navbar-inner .btn {

View File

@ -22,15 +22,15 @@
<i class="icon-refresh"></i>
</button></li>
<li class="btn-group"><button class="btn action-create-file"
title="New local file">
title="New local document">
<i class="icon-file"></i>
</button>
<button class="btn" title="Delete local file"
<button class="btn" title="Delete local document"
data-toggle="modal" data-target="#modal-remove-file-confirm">
<i class="icon-trash"></i>
</button>
<button class="btn dropdown-toggle" data-toggle="dropdown"
title="Open local file">
title="Open local document">
<i class="icon-folder-open"></i>
</button>
<ul id="file-selector" class="dropdown-menu">
@ -47,27 +47,27 @@
<li class="dropdown-submenu"><a href="#"><i
class="icon-gdrive"></i> Google Drive</a>
<ul class="dropdown-menu">
<li><a class="action-upload-gdrive" href="#">Export
to Google Drive</a></li>
<li><a href="#" data-toggle="modal" data-target="#modal-download-gdrive">Import
from Google Drive</a></li>
<li><a href="#" class="action-upload-gdrive">Export to
Google Drive</a></li>
<li><a href="#" data-toggle="modal"
data-target="#modal-download-gdrive">Import from Google
Drive</a></li>
</ul></li>
<li class="dropdown-submenu"><a href="#"><i
class="icon-dropbox"></i> Dropbox</a>
<ul class="dropdown-menu">
<li><a class="action-upload-dropbox" href="#">Export
to Dropbox</a></li>
<li><a class="action-download-dropbox" href="#">Import
from Dropbox</a></li>
<li><a class="action-upload-dropbox" href="#">Export to
Dropbox</a></li>
<li><a class="action-download-dropbox" href="#">Import
from Dropbox</a></li>
</ul></li>
<li><a href="#"
data-toggle="modal" data-target="#modal-manage-sync"><i
class="icon-refresh"></i> Manage synchronization</a></li>
<li><a href="#" data-toggle="modal"
data-target="#modal-manage-sync"><i class="icon-refresh"></i>
Manage synchronization</a></li>
<li class="divider"></li>
<li><a href="#"
data-toggle="modal" data-target="#modal-settings"
class="action-load-settings"><i class="icon-cog"></i>
Settings</a></li>
<li><a href="#" data-toggle="modal"
data-target="#modal-settings" class="action-load-settings"><i
class="icon-cog"></i> Settings</a></li>
<li><a href="#" data-toggle="modal"
data-target="#modal-about"><i class="icon-question-sign"></i>
About</a></li>
@ -76,9 +76,9 @@
<ul class="nav pull-right">
<li><i class="working-indicator icon-spinner hide"></i></li>
<li><a class="brand" id="file-title" href="#"
title="Rename current file"> </a></li>
title="Rename current document"> </a></li>
<li class="navbar-form"><input id="file-title-input"
type="text" class="span3 hide" placeholder="File title" /></li>
type="text" class="span3 hide" placeholder="Document title" /></li>
</ul>
</div>
</div>
@ -111,9 +111,13 @@
<h3>Import</h3>
</div>
<div class="modal-body">
<p>Please provide a Google Drive file ID:
</p>
<p><input type="text" class="border span6 gdrive-fileid" placeholder="File ID"></input></p>
<p>Please provide a Google Drive file ID:</p>
<div class="input-prepend">
<span class="add-on"><i class="icon-gdrive"></i></span><input
id="download-gdrive-fileid" type="text" class="span5" placeholder="Google Drive file ID"></input>
</div>
<p class="muted"><b>NOTE:</b> This will create a local copy of your Google Drive document
and keep it synchronized.</p>
</div>
<div class="modal-footer">
<a href="#" class="btn" data-dismiss="modal">Cancel</a> <a href="#"
@ -133,14 +137,20 @@
is synchronized with these locations:
</p>
</div>
<br class="msg-sync-list hide" />
<p class="msg-sync-list hide muted"><b>NOTE:</b> Removing a
synchronized location will not delete any file.</p>
<p class="msg-no-sync hide">"<span class="file-title"></span>" is
not synchronized.
</p>
<p class="msg-no-sync hide muted"><b>NOTE:</b> You can add
synchronized locations using the top-right menu.</p>
<p>Add a synchronized location manually:</p>
<div class="input-prepend input-append">
<span class="add-on"><i class="icon-gdrive"></i></span><input
id="manual-gdrive-fileid" type="text" class="span5" placeholder="Google Drive file ID"></input>
<a class="btn action-manual-gdrive" data-dismiss="modal"><i class="icon-ok"></i></a>
</div>
<p class="muted"><b>NOTE:</b> Adding a synchronized location will
first upload the local document and overwrite the existing file on the
server.</p>
</div>
<div class="modal-footer">
<a href="#" class="btn btn-primary" data-dismiss="modal">Close</a>
@ -178,69 +188,73 @@
aria-hidden="true">&times;</button>
<img src="img/stackedit-promo.png" />
</div>
<div class="modal-body">
<dl>
<dt>About:</dt>
<dd>
<a target="_blank" href="https://github.com/benweet/stackedit/">GitHub
page</a>
</dd>
<dd>
<a target="_blank"
href="https://chrome.google.com/webstore/detail/stackedit/iiooodelglhkcpgbajoejffhijaclcdg">Chrome
app</a>
</dd>
<dd>
<a target="_blank" href="https://twitter.com/stackedit/">Follow
on Twitter</a>
</dd>
<dd>
<a target="_blank" href="https://www.facebook.com/stackedit/">Follow
on Facebook</a>
</dd>
<dd>
<a target="_blank"
href="https://plus.google.com/b/110816046787593496375/110816046787593496375">Follow
on Google+</a>
</dd>
</dl>
<dl>
<dt>Developer:</dt>
<dd>
<a target="_blank" href="http://www.benoitschweblin.com">Benoit
Schweblin</a>
<dd>
</dl>
<dl>
<dt>Credits:</dt>
<dd>
<a target="_blank" href="http://jquery.com/">jQuery</a>
</dd>
<div class="modal-body">
<dl>
<dt>About:</dt>
<dd>
<a target="_blank" href="https://github.com/benweet/stackedit/">GitHub
page</a>
</dd>
<dd>
<a target="_blank"
href="https://chrome.google.com/webstore/detail/stackedit/iiooodelglhkcpgbajoejffhijaclcdg">Chrome
app</a>
</dd>
<dd>
<a target="_blank" href="https://twitter.com/stackedit/">Follow
on Twitter</a>
</dd>
<dd>
<a target="_blank" href="https://www.facebook.com/stackedit/">Follow
on Facebook</a>
</dd>
<dd>
<a target="_blank"
href="https://plus.google.com/b/110816046787593496375/110816046787593496375">Follow
on Google+</a>
</dd>
</dl>
<dl>
<dt>Developer:</dt>
<dd>
<a target="_blank" href="http://www.benoitschweblin.com">Benoit
Schweblin</a>
<dd>
</dl>
<dl>
<dt>Credit:</dt>
<dd>
<a target="_blank" href="http://twitter.github.com/bootstrap/">Bootstrap</a>
</dd>
<dd>
<a target="_blank" href="http://requirejs.org/">RequireJS</a>
</dd>
<dd>
<a target="_blank" href="https://code.google.com/p/pagedown/">PageDown</a>
</dd>
<dd>
<a target="_blank" href="http://layout.jquery-dev.net/">UI
Layout</a>
<a target="_blank" href="http://glyphicons.com/">GLYPHICONS</a>
</dd>
<dd>
<a target="_blank" href="https://github.com/stanlemon/jGrowl/">jGrowl</a>
</dd>
</dl>
<p>Copyright 2013 <a target="_blank"
href="http://www.benoitschweblin.com">Benoit Schweblin</a><br />
Licensed under an <a target="_blank"
href="http://www.apache.org/licenses/LICENSE-2.0">Apache License</a></p>
</div>
<dd>
<a target="_blank" href="http://jquery.com/">jQuery</a>
</dd>
<dd>
<a target="_blank" href="http://layout.jquery-dev.net/">jQuery UI
Layout</a>
</dd>
<dd>
<a target="_blank" href="https://code.google.com/p/pagedown/">PageDown</a>
</dd>
<dd>
<a target="_blank" href="http://requirejs.org/">RequireJS</a>
</dd>
</dl>
<p>Copyright 2013 <a target="_blank"
href="http://www.benoitschweblin.com">Benoit Schweblin</a><br />
Licensed under an <a target="_blank"
href="http://www.apache.org/licenses/LICENSE-2.0">Apache License</a></p>
</div>
<div class="modal-footer">
<a href="#" class="btn btn-primary" data-dismiss="modal">Close</a>
</div>
</div>
</body>
</html>

View File

@ -3,7 +3,7 @@ var GOOGLE_SCOPES = [ 'https://www.googleapis.com/auth/drive.install',
var DEFAULT_FILE_TITLE = "Title";
var GDRIVE_DEFAULT_FILE_TITLE = "New Markdown document";
var CHECK_ONLINE_PERIOD = 60000;
var AJAX_TIMEOUT = 5000;
var AJAX_TIMEOUT = 10000;
var ASYNC_TASK_DEFAULT_TIMEOUT = 30000;
var AUTH_POPUP_TIMEOUT = 90000;
var SYNC_PERIOD = 60000;

View File

@ -11,12 +11,31 @@ define(["jquery", "bootstrap", "jgrowl", "layout", "Markdown.Editor"], function(
// Usage: callback = callback || core.doNothing;
core.doNothing = function() {};
// Useful function
core.getInputValue = function(element, event) {
var value = element.val();
if (value !== undefined) {
value = value.replace(/^\s+|\s+$/g, '');
element.val(undefined);
}
if (value === undefined || value.length === 0) {
element.stop(true, true).addClass("error").delay(400)
.switchClass("error");
event.stopPropagation();
return undefined;
}
return value;
};
// Used by asyncTaskRunner
core.showWorkingIndicator = function(show) {
if (show === false) {
$(".working-indicator").addClass("hide");
$("body").removeClass("working");
} else {
$(".working-indicator").removeClass("hide");
$("body").addClass("working");
}
};
@ -58,7 +77,7 @@ define(["jquery", "bootstrap", "jgrowl", "layout", "Markdown.Editor"], function(
};
core.setOnline = function() {
if(offline === true) {
if(core.isOffline === true) {
$(".msg-offline").parents(".jGrowl-notification").trigger(
'jGrowl.beforeClose');
core.isOffline = false;
@ -117,8 +136,6 @@ define(["jquery", "bootstrap", "jgrowl", "layout", "Markdown.Editor"], function(
spacing_closed : 15,
togglerLength_open : 90,
togglerLength_closed : 90,
center__minWidth : 100,
center__minHeight : 100,
stateManagement__enabled : false
};
if (settings.layoutOrientation == "horizontal") {
@ -184,7 +201,6 @@ define(["jquery", "bootstrap", "jgrowl", "layout", "Markdown.Editor"], function(
$("#wmd-redo-button").append($("<i>").addClass("icon-share-alt"));
};
// Base64 conversion
core.encodeBase64 = function(str) {
if (str.length === 0) {

View File

@ -76,10 +76,15 @@ define(["jquery", "core", "gdrive", "synchronizer", "async-runner"], function($,
window.open(uriContent, 'file');
});
$(".action-upload-gdrive").click(uploadGdrive);
$(".action-download-gdrive").click(function() {
var fileId = $(".gdrive-fileid").val();
$(".gdrive-fileid").val("");
downloadGdrive(fileId);
$(".action-download-gdrive").click(function(event) {
var fileId = core.getInputValue($("#download-gdrive-fileid"), event);
if(checkGdriveFileId(fileId) === true) {
gdrive.importFiles([fileId]);
}
});
$(".action-manual-gdrive").click(function(event) {
var fileId = core.getInputValue($("#manual-gdrive-fileid"), event);
manualGdrive(fileId);
});
$(".action-download-dropbox").click(function() {
core.showMessage("Sorry, Dropbox synchronization is not yet available.");
@ -216,7 +221,6 @@ define(["jquery", "core", "gdrive", "synchronizer", "async-runner"], function($,
}
return result;
}
synchronizer.useGoogleDrive = useGoogleDrive;
// Update the file title
var title = localStorage[fileIndex + ".title"];
@ -243,6 +247,7 @@ define(["jquery", "core", "gdrive", "synchronizer", "async-runner"], function($,
}
$("#file-selector").append(li);
}
synchronizer.useGoogleDrive = useGoogleDrive;
};
// Remove a synchronized location
@ -280,22 +285,54 @@ define(["jquery", "core", "gdrive", "synchronizer", "async-runner"], function($,
var content = localStorage[fileIndex + ".content"];
var title = localStorage[fileIndex + ".title"];
gdrive.createFile(title, content, function(fileSyncIndex) {
if (fileSyncIndex) {
localStorage[fileIndex + ".sync"] += fileSyncIndex + ";";
refreshManageSync();
fileManager.updateFileTitles();
core.showMessage('"' + title
+ '" will now be synchronized on Google Drive.');
if (fileSyncIndex === undefined) {
return;
}
localStorage[fileIndex + ".sync"] += fileSyncIndex + ";";
refreshManageSync();
fileManager.updateFileTitles();
core.showMessage('"' + title
+ '" will now be synchronized on Google Drive.');
});
}
function downloadGdrive(fileId) {
fileId = fileId.trim();
if(fileId.length === 0) {
function checkGdriveFileId(fileId) {
if(!fileId) {
return false;
}
// Check that file is not synchronized with an other one
var fileSyncIndex = SYNC_PROVIDER_GDRIVE + fileId;
var fileIndex = fileManager.getFileIndexFromSync(fileSyncIndex);
if(fileIndex !== undefined) {
var title = localStorage[fileIndex + ".title"];
core.showError('Google Drive file is already synchronized with "' + title + '"');
return false;
}
return true;
}
function manualGdrive(fileId) {
if(checkGdriveFileId(fileId) === false) {
return;
}
gdrive.importFiles([fileId]);
var fileIndex = localStorage["file.current"];
var title = localStorage[fileIndex + ".title"];
gdrive.downloadMetadata([fileId], function(result) {
if(result === undefined || result.length === 0) {
return;
}
var file = result[0];
var fileSyncIndex = SYNC_PROVIDER_GDRIVE + file.id;
localStorage[fileSyncIndex + ".etag"] = file.etag;
localStorage[fileIndex + ".sync"] += fileSyncIndex + ";";
refreshManageSync();
fileManager.updateFileTitles();
core.showMessage('"' + title
+ '" will now be synchronized on Google Drive.');
// Force synchronization
synchronizer.addFileForUpload(fileIndex);
synchronizer.forceSync();
});
}
function refreshManageSync() {
@ -317,7 +354,7 @@ define(["jquery", "core", "gdrive", "synchronizer", "async-runner"], function($,
'<i class="icon-gdrive"></i>'));
line.append($("<input>").prop("type", "text").prop(
"disabled", true).addClass("span5").val(
"FileID="
"ID="
+ fileSyncIndex.substring(SYNC_PROVIDER_GDRIVE.length)));
}
line.append($("<a>").addClass("btn").html(

View File

@ -254,7 +254,7 @@ define(["jquery", "core", "async-runner"], function($, core, asyncTaskRunner) {
message: jqXHR.statusText
};
// Handle error
if(error.code === 403 || error.code === 404) {
if(error.code === 404) {
error = "File is not available.";
}
handleError(error, asyncTask, callback);
@ -337,8 +337,7 @@ define(["jquery", "core", "async-runner"], function($, core, asyncTaskRunner) {
if (typeof error === "string") {
errorMsg = error;
}
else if ((error.code >= 500 && error.code < 600) ||
(error.code === 401 && error.message == "Login Required")) { // Sometimes we have this 401 error
else if (error.code >= 500 && error.code < 600) {
errorMsg = "Google Drive is not accessible.";
// Retry as described in Google's best practices
asyncTask.retry();