Import images from Google+
This commit is contained in:
parent
200a99b8e9
commit
5b2531472c
@ -6,7 +6,7 @@ index.html
|
|||||||
viewer.html
|
viewer.html
|
||||||
css/main-min.css
|
css/main-min.css
|
||||||
js/main-min.js
|
js/main-min.js
|
||||||
js/lib/require.js
|
js/libs/require.js
|
||||||
img/ajax-loader.gif
|
img/ajax-loader.gif
|
||||||
img/glyphicons-halflings.png
|
img/glyphicons-halflings.png
|
||||||
img/glyphicons-halflings-white.png
|
img/glyphicons-halflings-white.png
|
||||||
|
@ -85,13 +85,29 @@ input[type="color"]:focus,
|
|||||||
box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(128, 128, 128, 0.6) !important;
|
box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(128, 128, 128, 0.6) !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
input:-moz-placeholder,
|
||||||
|
textarea:-moz-placeholder {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
input:-ms-input-placeholder,
|
||||||
|
textarea:-ms-input-placeholder {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
input::-webkit-input-placeholder,
|
||||||
|
textarea::-webkit-input-placeholder {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
.help-block {
|
.help-block {
|
||||||
color: #999999;
|
color: #999999;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
line-height: 17px;
|
line-height: 17px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.error {
|
.modal textarea.error,
|
||||||
|
.modal input.error {
|
||||||
border-color: #ff8661 !important;
|
border-color: #ff8661 !important;
|
||||||
-webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(255, 134, 97, 0.6) !important;
|
-webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(255, 134, 97, 0.6) !important;
|
||||||
-moz-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(255, 134, 97, 0.6) !important;
|
-moz-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(255, 134, 97, 0.6) !important;
|
||||||
@ -273,7 +289,7 @@ hr {
|
|||||||
border: none !important;
|
border: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
#wmd-preview {
|
.preview-container {
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -433,6 +449,13 @@ div.dropdown-menu i {
|
|||||||
background-position: -127px 0;
|
background-position: -127px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.icon-gplus {
|
||||||
|
background-image: url("../img/icons.png") !important;
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
background-position: -145px 0;
|
||||||
|
}
|
||||||
|
|
||||||
.working-indicator {
|
.working-indicator {
|
||||||
background-image: none !important;
|
background-image: none !important;
|
||||||
width: 43px;
|
width: 43px;
|
||||||
@ -490,6 +513,10 @@ div.dropdown-menu i {
|
|||||||
z-index: 1050 !important;
|
z-index: 1050 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.action-import-image-gplus {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
#modal-settings .modal-header {
|
#modal-settings .modal-header {
|
||||||
padding-bottom: 0;
|
padding-bottom: 0;
|
||||||
}
|
}
|
||||||
|
BIN
img/icons.png
BIN
img/icons.png
Binary file not shown.
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 4.2 KiB |
54
index.html
54
index.html
@ -24,7 +24,7 @@
|
|||||||
var require = { baseUrl : "js", deps : [ "main" + suffix ] };
|
var require = { baseUrl : "js", deps : [ "main" + suffix ] };
|
||||||
var viewerMode = false;
|
var viewerMode = false;
|
||||||
</script>
|
</script>
|
||||||
<script src="js/lib/require.js"></script>
|
<script src="js/libs/require.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="navbar" class="navbar navbar-fixed-top ui-layout-north">
|
<div id="navbar" class="navbar navbar-fixed-top ui-layout-north">
|
||||||
@ -116,8 +116,8 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<textarea id="wmd-input" class="ui-layout-center hide"></textarea>
|
<textarea id="wmd-input" class="ui-layout-center hide"></textarea>
|
||||||
<div class="ui-layout-east hide"></div>
|
<div class="ui-layout-east preview-container hide"></div>
|
||||||
<div class="ui-layout-south hide"></div>
|
<div class="ui-layout-south preview-container hide"></div>
|
||||||
|
|
||||||
<div id="modal-insert-link" class="modal hide">
|
<div id="modal-insert-link" class="modal hide">
|
||||||
<div class="modal-header">
|
<div class="modal-header">
|
||||||
@ -134,7 +134,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
<a href="#" class="btn action-close-insert-link" data-dismiss="modal">Cancel</a>
|
<a href="#" class="btn" data-dismiss="modal">Cancel</a>
|
||||||
<a href="#" class="btn btn-primary action-insert-link"
|
<a href="#" class="btn btn-primary action-insert-link"
|
||||||
data-dismiss="modal">OK</a>
|
data-dismiss="modal">OK</a>
|
||||||
</div>
|
</div>
|
||||||
@ -155,12 +155,49 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
<a href="#" class="btn action-close-insert-link" data-dismiss="modal">Cancel</a>
|
<a href="#" class="btn action-import-image-gplus" data-dismiss="modal"><i class="icon-gplus"></i> Import from Google+</a>
|
||||||
|
<a href="#" class="btn" data-dismiss="modal">Cancel</a>
|
||||||
<a href="#" class="btn btn-primary action-insert-image"
|
<a href="#" class="btn btn-primary action-insert-image"
|
||||||
data-dismiss="modal">OK</a>
|
data-dismiss="modal">OK</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div id="modal-import-image" class="modal hide">
|
||||||
|
<div class="modal-header">
|
||||||
|
<button type="button" class="close action-close-insert-link"
|
||||||
|
data-dismiss="modal" aria-hidden="true">×</button>
|
||||||
|
<h3>Import image</h3>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<div class="form-horizontal">
|
||||||
|
<div class="control-group">
|
||||||
|
<div class="controls">
|
||||||
|
<img>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label" for="input-import-image-title">Title (optional)</label>
|
||||||
|
<div class="controls">
|
||||||
|
<input type="text" id="input-import-image-title"
|
||||||
|
placeholder="Image title">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="control-group">
|
||||||
|
<label class="control-label" for="input-import-image-size">Size limit
|
||||||
|
(optional)</label>
|
||||||
|
<div class="controls">
|
||||||
|
<input type="text" id="input-import-image-size" placeholder="345" class="input-mini"><span class="help-inline">px</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<a href="#" class="btn" data-dismiss="modal">Cancel</a>
|
||||||
|
<a href="#" class="btn btn-primary action-import-image"
|
||||||
|
data-dismiss="modal">OK</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div id="modal-remove-file-confirm" class="modal hide">
|
<div id="modal-remove-file-confirm" class="modal hide">
|
||||||
<div class="modal-header">
|
<div class="modal-header">
|
||||||
<button type="button" class="close" data-dismiss="modal"
|
<button type="button" class="close" data-dismiss="modal"
|
||||||
@ -269,14 +306,14 @@
|
|||||||
"<span class="file-title"></span>" is not synchronized yet.
|
"<span class="file-title"></span>" is not synchronized yet.
|
||||||
</blockquote>
|
</blockquote>
|
||||||
<p>Add a synchronized location manually:</p>
|
<p>Add a synchronized location manually:</p>
|
||||||
<div class="input-prepend input-append">
|
<div class="input-prepend input-append sync-manual">
|
||||||
<span class="add-on" title="Google Drive"><i
|
<span class="add-on" title="Google Drive"><i
|
||||||
class="icon-gdrive"></i></span><input id="input-sync-manual-gdrive-id"
|
class="icon-gdrive"></i></span><input id="input-sync-manual-gdrive-id"
|
||||||
type="text" class="span5" placeholder="GoogleDriveFileID"></input>
|
type="text" class="span5" placeholder="GoogleDriveFileID"></input>
|
||||||
<a class="btn action-sync-manual-gdrive" title="Add location"
|
<a class="btn action-sync-manual-gdrive" title="Add location"
|
||||||
data-dismiss="modal"><i class="icon-ok"></i></a>
|
data-dismiss="modal"><i class="icon-ok"></i></a>
|
||||||
</div>
|
</div>
|
||||||
<div class="input-prepend input-append">
|
<div class="input-prepend input-append sync-manual">
|
||||||
<span class="add-on" title="Dropbox"><i class="icon-dropbox"></i></span><input
|
<span class="add-on" title="Dropbox"><i class="icon-dropbox"></i></span><input
|
||||||
id="input-sync-manual-dropbox-path" type="text" class="span5"
|
id="input-sync-manual-dropbox-path" type="text" class="span5"
|
||||||
placeholder="/dropbox/file/path"></input> <a
|
placeholder="/dropbox/file/path"></input> <a
|
||||||
@ -718,6 +755,9 @@
|
|||||||
<dd>
|
<dd>
|
||||||
<a target="_blank" href="http://underscorejs.org/">Underscore.js</a>
|
<a target="_blank" href="http://underscorejs.org/">Underscore.js</a>
|
||||||
</dd>
|
</dd>
|
||||||
|
<dd>
|
||||||
|
<a target="_blank" href="https://github.com/alexanderdickson/waitForImages">waitForImages</a>
|
||||||
|
</dd>
|
||||||
</dl>
|
</dl>
|
||||||
<dl>
|
<dl>
|
||||||
<dt>Related projects:</dt>
|
<dt>Related projects:</dt>
|
||||||
|
@ -14,7 +14,7 @@ var GDRIVE_DEFAULT_FILE_TITLE = "New Markdown document";
|
|||||||
var CHECK_ONLINE_PERIOD = 120000;
|
var CHECK_ONLINE_PERIOD = 120000;
|
||||||
var AJAX_TIMEOUT = 30000;
|
var AJAX_TIMEOUT = 30000;
|
||||||
var ASYNC_TASK_DEFAULT_TIMEOUT = 60000;
|
var ASYNC_TASK_DEFAULT_TIMEOUT = 60000;
|
||||||
var ASYNC_TASK_LONG_TIMEOUT = 120000;
|
var ASYNC_TASK_LONG_TIMEOUT = 180000;
|
||||||
var SYNC_PERIOD = 180000;
|
var SYNC_PERIOD = 180000;
|
||||||
var USER_IDLE_THRESHOLD = 300000;
|
var USER_IDLE_THRESHOLD = 300000;
|
||||||
var TEMPORARY_FILE_INDEX = "file.tempIndex";
|
var TEMPORARY_FILE_INDEX = "file.tempIndex";
|
||||||
|
89
js/core.js
89
js/core.js
@ -6,9 +6,9 @@ define([
|
|||||||
"extension-manager",
|
"extension-manager",
|
||||||
"storage",
|
"storage",
|
||||||
"config",
|
"config",
|
||||||
"lib/bootstrap",
|
"libs/bootstrap",
|
||||||
"lib/layout",
|
"libs/layout",
|
||||||
"lib/Markdown.Editor"
|
"libs/Markdown.Editor"
|
||||||
], function($, _, utils, settings, extensionMgr) {
|
], function($, _, utils, settings, extensionMgr) {
|
||||||
|
|
||||||
var core = {};
|
var core = {};
|
||||||
@ -172,7 +172,7 @@ define([
|
|||||||
extensionMgr.onLayoutConfigure(layoutGlobalConfig);
|
extensionMgr.onLayoutConfigure(layoutGlobalConfig);
|
||||||
if(settings.layoutOrientation == "horizontal") {
|
if(settings.layoutOrientation == "horizontal") {
|
||||||
$(".ui-layout-south").remove();
|
$(".ui-layout-south").remove();
|
||||||
$(".ui-layout-east").addClass("well").prop("id", "wmd-preview");
|
$(".preview-container").html('<div id="wmd-preview" class="well"></div>');
|
||||||
layout = $('body').layout($.extend(layoutGlobalConfig, {
|
layout = $('body').layout($.extend(layoutGlobalConfig, {
|
||||||
east__resizable: true,
|
east__resizable: true,
|
||||||
east__size: .5,
|
east__size: .5,
|
||||||
@ -181,7 +181,7 @@ define([
|
|||||||
}
|
}
|
||||||
else if(settings.layoutOrientation == "vertical") {
|
else if(settings.layoutOrientation == "vertical") {
|
||||||
$(".ui-layout-east").remove();
|
$(".ui-layout-east").remove();
|
||||||
$(".ui-layout-south").addClass("well").prop("id", "wmd-preview");
|
$(".preview-container").html('<div id="wmd-preview" class="well"></div>');
|
||||||
layout = $('body').layout($.extend(layoutGlobalConfig, {
|
layout = $('body').layout($.extend(layoutGlobalConfig, {
|
||||||
south__resizable: true,
|
south__resizable: true,
|
||||||
south__size: .5,
|
south__size: .5,
|
||||||
@ -199,32 +199,33 @@ define([
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Create the PageDown editor
|
// Create the PageDown editor
|
||||||
var insertLinkCallback = undefined;
|
var editor = undefined;
|
||||||
|
var documentContent = undefined;
|
||||||
core.createEditor = function(onTextChange) {
|
core.createEditor = function(onTextChange) {
|
||||||
|
documentContent = undefined;
|
||||||
|
$("#wmd-input, .preview-container").scrollTop(0);
|
||||||
|
if(editor !== undefined) {
|
||||||
|
// Only restart if the editor is already created
|
||||||
|
editor.restart();
|
||||||
|
return;
|
||||||
|
}
|
||||||
var converter = new Markdown.Converter();
|
var converter = new Markdown.Converter();
|
||||||
var editor = new Markdown.Editor(converter);
|
editor = new Markdown.Editor(converter);
|
||||||
// Custom insert link dialog
|
// Custom insert link dialog
|
||||||
editor.hooks.set("insertLinkDialog", function(callback) {
|
editor.hooks.set("insertLinkDialog", function(callback) {
|
||||||
insertLinkCallback = callback;
|
core.insertLinkCallback = callback;
|
||||||
utils.resetModalInputs();
|
utils.resetModalInputs();
|
||||||
$("#modal-insert-link").modal();
|
$("#modal-insert-link").modal();
|
||||||
_.defer(function() {
|
|
||||||
$("#input-insert-link").focus();
|
|
||||||
});
|
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
// Custom insert image dialog
|
// Custom insert image dialog
|
||||||
editor.hooks.set("insertImageDialog", function(callback) {
|
editor.hooks.set("insertImageDialog", function(callback) {
|
||||||
insertLinkCallback = callback;
|
core.insertLinkCallback = callback;
|
||||||
utils.resetModalInputs();
|
utils.resetModalInputs();
|
||||||
$("#modal-insert-image").modal();
|
$("#modal-insert-image").modal();
|
||||||
_.defer(function() {
|
|
||||||
$("#input-insert-image").focus();
|
|
||||||
});
|
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
|
|
||||||
var documentContent = undefined;
|
|
||||||
function checkDocumentChanges() {
|
function checkDocumentChanges() {
|
||||||
var newDocumentContent = $("#wmd-input").val();
|
var newDocumentContent = $("#wmd-input").val();
|
||||||
if(documentContent !== undefined && documentContent != newDocumentContent) {
|
if(documentContent !== undefined && documentContent != newDocumentContent) {
|
||||||
@ -232,7 +233,7 @@ define([
|
|||||||
}
|
}
|
||||||
documentContent = newDocumentContent;
|
documentContent = newDocumentContent;
|
||||||
}
|
}
|
||||||
var previewWrapper = undefined;
|
var previewWrapper;
|
||||||
if(settings.lazyRendering === true) {
|
if(settings.lazyRendering === true) {
|
||||||
previewWrapper = function(makePreview) {
|
previewWrapper = function(makePreview) {
|
||||||
var debouncedMakePreview = _.debounce(makePreview, 500);
|
var debouncedMakePreview = _.debounce(makePreview, 500);
|
||||||
@ -255,13 +256,9 @@ define([
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
editor.hooks.chain("onPreviewRefresh", extensionMgr.onAsyncPreview);
|
|
||||||
extensionMgr.onEditorConfigure(editor);
|
extensionMgr.onEditorConfigure(editor);
|
||||||
|
editor.hooks.chain("onPreviewRefresh", extensionMgr.onAsyncPreview);
|
||||||
$("#wmd-input, #wmd-preview").scrollTop(0);
|
|
||||||
$("#wmd-button-bar").empty();
|
|
||||||
editor.run(previewWrapper);
|
editor.run(previewWrapper);
|
||||||
firstChange = false;
|
|
||||||
|
|
||||||
// Hide default buttons
|
// Hide default buttons
|
||||||
$(".wmd-button-row").addClass("btn-group").find("li:not(.wmd-spacer)").addClass("btn").css("left", 0).find("span").hide();
|
$(".wmd-button-row").addClass("btn-group").find("li:not(.wmd-spacer)").addClass("btn").css("left", 0).find("span").hide();
|
||||||
@ -324,21 +321,44 @@ define([
|
|||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
$(".modal").on('shown', function() {
|
||||||
|
// Focus on the first input when modal opens
|
||||||
|
_.defer(function(elt) {
|
||||||
|
elt.find("input:enabled:visible:first").focus();
|
||||||
|
}, $(this));
|
||||||
|
}).on('hidden', function() {
|
||||||
|
// Focus on the editor when modal is gone
|
||||||
|
if($(this).is(":hidden")) {
|
||||||
|
_.defer(function() {
|
||||||
|
$("#wmd-input").focus();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}).keyup(function(e) {
|
||||||
|
// Handle enter key in modals
|
||||||
|
if(e.which == 13 && !$(e.target).is("textarea")) {
|
||||||
|
$(this).find(".modal-footer a:last").click();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Click events on "insert link" and "insert image" dialog buttons
|
// Click events on "insert link" and "insert image" dialog buttons
|
||||||
$(".action-insert-link").click(function(e) {
|
$(".action-insert-link").click(function(e) {
|
||||||
var value = utils.getInputTextValue($("#input-insert-link"), e);
|
var value = utils.getInputTextValue($("#input-insert-link"), e);
|
||||||
if(value !== undefined) {
|
if(value !== undefined) {
|
||||||
insertLinkCallback(value);
|
core.insertLinkCallback(value);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
$(".action-insert-image").click(function(e) {
|
$(".action-insert-image").click(function(e) {
|
||||||
var value = utils.getInputTextValue($("#input-insert-image"), e);
|
var value = utils.getInputTextValue($("#input-insert-image"), e);
|
||||||
if(value !== undefined) {
|
if(value !== undefined) {
|
||||||
insertLinkCallback(value);
|
core.insertLinkCallback(value);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
$(".action-close-insert-link").click(function(e) {
|
|
||||||
insertLinkCallback(null);
|
// Hide events on "insert link" and "insert image" dialogs
|
||||||
|
$("#modal-insert-link, #modal-insert-image").on('hidden', function() {
|
||||||
|
if(core.insertLinkCallback !== undefined) {
|
||||||
|
core.insertLinkCallback(null);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Settings loading/saving
|
// Settings loading/saving
|
||||||
@ -374,7 +394,7 @@ define([
|
|||||||
"line-height": Math.round(settings.editorFontSize * (20 / 14)) + "px"
|
"line-height": Math.round(settings.editorFontSize * (20 / 14)) + "px"
|
||||||
});
|
});
|
||||||
|
|
||||||
// Manage tab key
|
// Handle tab key
|
||||||
$("#wmd-input").keydown(function(e) {
|
$("#wmd-input").keydown(function(e) {
|
||||||
if(e.keyCode === 9) {
|
if(e.keyCode === 9) {
|
||||||
var value = $(this).val();
|
var value = $(this).val();
|
||||||
@ -394,12 +414,14 @@ define([
|
|||||||
$(".tooltip-lazy-rendering").tooltip({
|
$(".tooltip-lazy-rendering").tooltip({
|
||||||
container: '#modal-settings',
|
container: '#modal-settings',
|
||||||
placement: 'right',
|
placement: 'right',
|
||||||
|
trigger: 'hover',
|
||||||
title: 'Disable preview rendering while typing in order to offload CPU. Refresh preview after 500 ms of inactivity.'
|
title: 'Disable preview rendering while typing in order to offload CPU. Refresh preview after 500 ms of inactivity.'
|
||||||
});
|
});
|
||||||
$(".tooltip-default-content").tooltip({
|
$(".tooltip-default-content").tooltip({
|
||||||
html: true,
|
html: true,
|
||||||
container: '#modal-settings',
|
container: '#modal-settings',
|
||||||
placement: 'right',
|
placement: 'right',
|
||||||
|
trigger: 'hover',
|
||||||
title: 'Thanks for supporting StackEdit by adding a backlink in your documents!'
|
title: 'Thanks for supporting StackEdit by adding a backlink in your documents!'
|
||||||
});
|
});
|
||||||
$(".tooltip-template").tooltip({
|
$(".tooltip-template").tooltip({
|
||||||
@ -426,13 +448,13 @@ define([
|
|||||||
].join("")
|
].join("")
|
||||||
}).click(function(e) {
|
}).click(function(e) {
|
||||||
$(this).tooltip('show');
|
$(this).tooltip('show');
|
||||||
|
$(document).on("click.tooltip-template", function(e) {
|
||||||
|
$(".tooltip-template").tooltip('hide');
|
||||||
|
$(document).off("click.tooltip-template");
|
||||||
|
});
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
});
|
});
|
||||||
|
|
||||||
$(document).click(function(e) {
|
|
||||||
$(".tooltip-template").tooltip('hide');
|
|
||||||
});
|
|
||||||
|
|
||||||
// Reset inputs
|
// Reset inputs
|
||||||
$(".action-reset-input").click(function() {
|
$(".action-reset-input").click(function() {
|
||||||
utils.resetModalInputs();
|
utils.resetModalInputs();
|
||||||
@ -449,6 +471,11 @@ define([
|
|||||||
checkOnline();
|
checkOnline();
|
||||||
}
|
}
|
||||||
}, 1000);
|
}, 1000);
|
||||||
|
|
||||||
|
// Focus on the editor at startup
|
||||||
|
_.defer(function() {
|
||||||
|
$("#wmd-input").focus();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
return core;
|
return core;
|
||||||
|
@ -18,7 +18,8 @@ define([
|
|||||||
"extensions/math-jax",
|
"extensions/math-jax",
|
||||||
"extensions/email-converter",
|
"extensions/email-converter",
|
||||||
"extensions/scroll-link",
|
"extensions/scroll-link",
|
||||||
"lib/bootstrap"
|
"libs/bootstrap",
|
||||||
|
"libs/jquery.waitforimages"
|
||||||
], function($, _, utils, settings) {
|
], function($, _, utils, settings) {
|
||||||
|
|
||||||
var extensionMgr = {};
|
var extensionMgr = {};
|
||||||
@ -120,22 +121,22 @@ define([
|
|||||||
|
|
||||||
var onPreviewFinished = createHook("onPreviewFinished");
|
var onPreviewFinished = createHook("onPreviewFinished");
|
||||||
var onAsyncPreviewCallbackList = getExtensionCallbackList("onAsyncPreview");
|
var onAsyncPreviewCallbackList = getExtensionCallbackList("onAsyncPreview");
|
||||||
|
// The number of times we expect tryFinished to be called
|
||||||
|
var nbAsyncPreviewCallback = onAsyncPreviewCallbackList.length + 1;
|
||||||
extensionMgr["onAsyncPreview"] = function() {
|
extensionMgr["onAsyncPreview"] = function() {
|
||||||
logger.debug("onAsyncPreview");
|
logger.debug("onAsyncPreview");
|
||||||
// Call onPreviewFinished callbacks when all async preview are finished
|
// Call onPreviewFinished callbacks when all async preview are finished
|
||||||
var counter = 0;
|
var counter = 0;
|
||||||
function tryFinished() {
|
function tryFinished() {
|
||||||
if(counter === onAsyncPreviewCallbackList.length) {
|
if(++counter === nbAsyncPreviewCallback) {
|
||||||
onPreviewFinished();
|
onPreviewFinished();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// We assume images are loading in the preview
|
||||||
|
$("#wmd-preview").waitForImages(tryFinished);
|
||||||
_.each(onAsyncPreviewCallbackList, function(asyncPreviewCallback) {
|
_.each(onAsyncPreviewCallbackList, function(asyncPreviewCallback) {
|
||||||
asyncPreviewCallback(function() {
|
asyncPreviewCallback(tryFinished);
|
||||||
counter++;
|
|
||||||
tryFinished();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
tryFinished();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var accordionTmpl = [
|
var accordionTmpl = [
|
||||||
|
@ -9,10 +9,10 @@ define([
|
|||||||
extensionName: 'Button "Statistics"',
|
extensionName: 'Button "Statistics"',
|
||||||
optional: true,
|
optional: true,
|
||||||
defaultConfig: {
|
defaultConfig: {
|
||||||
name1: "Words",
|
name1: "Characters",
|
||||||
value1: "\\S+",
|
value1: "\\S",
|
||||||
name2: "Characters",
|
name2: "Words",
|
||||||
value2: "\\S",
|
value2: "\\S+",
|
||||||
name3: "Paragraphs",
|
name3: "Paragraphs",
|
||||||
value3: "\\S.*",
|
value3: "\\S.*",
|
||||||
},
|
},
|
||||||
|
@ -88,14 +88,24 @@ define([
|
|||||||
}
|
}
|
||||||
|
|
||||||
documentSelector.onReady = function() {
|
documentSelector.onReady = function() {
|
||||||
|
$("#file-selector").click(function() {
|
||||||
|
_.defer(function() {
|
||||||
|
$("#wmd-input").focus();
|
||||||
|
});
|
||||||
|
});
|
||||||
$(".action-open-file").click(function() {
|
$(".action-open-file").click(function() {
|
||||||
filterFileSelector();
|
filterFileSelector();
|
||||||
_.defer(function() {
|
_.defer(function() {
|
||||||
$("#file-search").val("").focus();
|
$("#file-search").val("").focus();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
$("#file-search").keyup(function() {
|
$("#file-search").keyup(function(e) {
|
||||||
filterFileSelector($(this).val());
|
if(e.which == 13 || e.which == 27) {
|
||||||
|
$(this).parent().click();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
filterFileSelector($(this).val());
|
||||||
|
}
|
||||||
}).click(function(event) {
|
}).click(function(event) {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
});
|
});
|
||||||
|
@ -59,6 +59,19 @@ define([
|
|||||||
manageSynchronization.onSyncExportSuccess = refreshDialog;
|
manageSynchronization.onSyncExportSuccess = refreshDialog;
|
||||||
manageSynchronization.onSyncRemoved = refreshDialog;
|
manageSynchronization.onSyncRemoved = refreshDialog;
|
||||||
|
|
||||||
|
manageSynchronization.onReady = function() {
|
||||||
|
// Handle enter key in the sync manual inputs
|
||||||
|
$(".sync-manual").each(function() {
|
||||||
|
var elt = $(this);
|
||||||
|
elt.find("input").keyup(function(e) {
|
||||||
|
if(e.which == 13) {
|
||||||
|
elt.find("a").click();
|
||||||
|
e.stopPropagation();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
return manageSynchronization;
|
return manageSynchronization;
|
||||||
|
|
||||||
});
|
});
|
@ -1,6 +1,6 @@
|
|||||||
define([
|
define([
|
||||||
"utils",
|
"utils",
|
||||||
"lib/Markdown.Extra"
|
"libs/Markdown.Extra"
|
||||||
], function(utils) {
|
], function(utils) {
|
||||||
|
|
||||||
var markdownExtra = {
|
var markdownExtra = {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
define([
|
define([
|
||||||
"lib/MathJax"
|
"libs/MathJax"
|
||||||
], function() {
|
], function() {
|
||||||
|
|
||||||
var mathJax = {
|
var mathJax = {
|
||||||
@ -175,7 +175,9 @@ define([
|
|||||||
// if we haven't done that already.
|
// if we haven't done that already.
|
||||||
//
|
//
|
||||||
function UpdateMJ() {
|
function UpdateMJ() {
|
||||||
if (!pending && ready) {
|
// sometimes ready flag is not set already (on opera basically)
|
||||||
|
//if (!pending && ready) {
|
||||||
|
if (!pending) {
|
||||||
pending = true;
|
pending = true;
|
||||||
HUB.Cancel();
|
HUB.Cancel();
|
||||||
HUB.Queue(RestartMJ);
|
HUB.Queue(RestartMJ);
|
||||||
@ -194,10 +196,10 @@ define([
|
|||||||
var converterObject = editorObject.getConverter();
|
var converterObject = editorObject.getConverter();
|
||||||
converterObject.hooks.chain("preConversion", removeMath);
|
converterObject.hooks.chain("preConversion", removeMath);
|
||||||
converterObject.hooks.chain("postConversion", replaceMath);
|
converterObject.hooks.chain("postConversion", replaceMath);
|
||||||
editorObject.hooks.chain("onPreviewRefresh", UpdateMJ);
|
|
||||||
};
|
};
|
||||||
mathJax.onAsyncPreview = function(callback) {
|
mathJax.onAsyncPreview = function(callback) {
|
||||||
afterRefreshCallback = callback;
|
afterRefreshCallback = callback;
|
||||||
|
UpdateMJ();
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
define([
|
define([
|
||||||
"jquery",
|
"jquery",
|
||||||
"underscore",
|
"underscore",
|
||||||
"lib/css_browser_selector"
|
"libs/css_browser_selector"
|
||||||
], function($, _) {
|
], function($, _) {
|
||||||
|
|
||||||
var scrollLink = {
|
var scrollLink = {
|
||||||
@ -10,8 +10,8 @@ define([
|
|||||||
optional: true,
|
optional: true,
|
||||||
settingsBloc: [
|
settingsBloc: [
|
||||||
'<p>Binds together editor and preview scrollbars.</p>',
|
'<p>Binds together editor and preview scrollbars.</p>',
|
||||||
'<blockquote class="muted"><b>NOTE:</b> ',
|
'<blockquote class="muted"><b>NOTE:</b>',
|
||||||
' The mapping between Markdown and HTML is based on the position of the title elements (h1, h2, ...) in the page. ',
|
' The mapping between Markdown and HTML is based on the position of the title elements (h1, h2, ...) in the page.',
|
||||||
' Therefore, if your document does not contain any title, the mapping will be linear and consequently less accurate.',
|
' Therefore, if your document does not contain any title, the mapping will be linear and consequently less accurate.',
|
||||||
'</bloquote>'
|
'</bloquote>'
|
||||||
].join("")
|
].join("")
|
||||||
@ -70,12 +70,12 @@ define([
|
|||||||
addMdSection(text.substring(offset, text.length - 2));
|
addMdSection(text.substring(offset, text.length - 2));
|
||||||
|
|
||||||
// Try to find corresponding sections in the preview
|
// Try to find corresponding sections in the preview
|
||||||
var previewElt = $("#wmd-preview");
|
var previewElt = $(".preview-container");
|
||||||
htmlSectionList = [];
|
htmlSectionList = [];
|
||||||
var htmlSectionOffset = 0;
|
var htmlSectionOffset = 0;
|
||||||
var previewScrollTop = previewElt.scrollTop();
|
var previewScrollTop = previewElt.scrollTop();
|
||||||
// Each title element is a section separator
|
// Each title element is a section separator
|
||||||
previewElt.children("h1,h2,h3,h4,h5,h6").each(function() {
|
$("#wmd-preview").children("h1,h2,h3,h4,h5,h6").each(function() {
|
||||||
// Consider div scroll position and header element top margin
|
// Consider div scroll position and header element top margin
|
||||||
var newSectionOffset = $(this).position().top + previewScrollTop + pxToFloat($(this).css('margin-top'));
|
var newSectionOffset = $(this).position().top + previewScrollTop + pxToFloat($(this).css('margin-top'));
|
||||||
htmlSectionList.push({
|
htmlSectionList.push({
|
||||||
@ -95,7 +95,6 @@ define([
|
|||||||
|
|
||||||
// apply Scroll Link
|
// apply Scroll Link
|
||||||
lastEditorScrollTop = -10;
|
lastEditorScrollTop = -10;
|
||||||
skipScrollLink = false;
|
|
||||||
isScrollPreview = false;
|
isScrollPreview = false;
|
||||||
runScrollLink();
|
runScrollLink();
|
||||||
}, 500);
|
}, 500);
|
||||||
@ -103,15 +102,14 @@ define([
|
|||||||
// -10 to be sure the gap is > 9
|
// -10 to be sure the gap is > 9
|
||||||
var lastEditorScrollTop = -10;
|
var lastEditorScrollTop = -10;
|
||||||
var lastPreviewScrollTop = -10;
|
var lastPreviewScrollTop = -10;
|
||||||
var skipScrollLink = false;
|
|
||||||
var isScrollPreview = false;
|
var isScrollPreview = false;
|
||||||
var runScrollLink = _.debounce(function() {
|
var runScrollLink = _.debounce(function() {
|
||||||
if(skipScrollLink === true || mdSectionList.length === 0 || mdSectionList.length !== htmlSectionList.length) {
|
if(mdSectionList.length === 0 || mdSectionList.length !== htmlSectionList.length) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
var editorElt = $("#wmd-input");
|
var editorElt = $("#wmd-input");
|
||||||
var editorScrollTop = editorElt.scrollTop();
|
var editorScrollTop = editorElt.scrollTop();
|
||||||
var previewElt = $("#wmd-preview");
|
var previewElt = $(".preview-container");
|
||||||
var previewScrollTop = previewElt.scrollTop();
|
var previewScrollTop = previewElt.scrollTop();
|
||||||
function animate(srcScrollTop, srcSectionList, destElt, destSectionList, lastDestScrollTop, callback) {
|
function animate(srcScrollTop, srcSectionList, destElt, destSectionList, lastDestScrollTop, callback) {
|
||||||
// Find the section corresponding to the offset
|
// Find the section corresponding to the offset
|
||||||
@ -163,7 +161,7 @@ define([
|
|||||||
};
|
};
|
||||||
|
|
||||||
scrollLink.onLayoutCreated = function() {
|
scrollLink.onLayoutCreated = function() {
|
||||||
$("#wmd-preview").scroll(function() {
|
$(".preview-container").scroll(function() {
|
||||||
isScrollPreview = true;
|
isScrollPreview = true;
|
||||||
runScrollLink();
|
runScrollLink();
|
||||||
});
|
});
|
||||||
@ -174,27 +172,21 @@ define([
|
|||||||
};
|
};
|
||||||
|
|
||||||
scrollLink.onEditorConfigure = function(editor) {
|
scrollLink.onEditorConfigure = function(editor) {
|
||||||
skipScrollLink = true;
|
|
||||||
lastPreviewScrollTop = 0;
|
lastPreviewScrollTop = 0;
|
||||||
editor.hooks.chain("onPreviewRefresh", function() {
|
editor.getConverter().hooks.chain("postConversion", function(text) {
|
||||||
skipScrollLink = true;
|
// To avoid losing scrolling position before elements are fully loaded
|
||||||
|
$("#wmd-preview").height($("#wmd-preview").height());
|
||||||
|
return text;
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
scrollLink.onPreviewFinished = function() {
|
scrollLink.onPreviewFinished = function() {
|
||||||
// MathJax may have change the scrolling position. Restore it.
|
// Now set the correct height
|
||||||
if(lastPreviewScrollTop >= 0) {
|
$("#wmd-preview").height("auto");
|
||||||
$("#wmd-preview").scrollTop(lastPreviewScrollTop);
|
|
||||||
}
|
|
||||||
_.defer(function() {
|
_.defer(function() {
|
||||||
// Modify scroll position of the preview not the editor
|
// Modify scroll position of the preview not the editor
|
||||||
lastEditorScrollTop = -10;
|
lastEditorScrollTop = -10;
|
||||||
buildSections();
|
buildSections();
|
||||||
// Preview may change if images are loading
|
|
||||||
$("#wmd-preview img").load(function() {
|
|
||||||
lastEditorScrollTop = -10;
|
|
||||||
buildSections();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -6,7 +6,7 @@ define([
|
|||||||
"settings",
|
"settings",
|
||||||
"extension-manager",
|
"extension-manager",
|
||||||
"file-system",
|
"file-system",
|
||||||
"lib/text!../WELCOME.md"
|
"libs/text!../WELCOME.md"
|
||||||
], function($, _, core, utils, settings, extensionMgr, fileSystem, welcomeContent) {
|
], function($, _, core, utils, settings, extensionMgr, fileSystem, welcomeContent) {
|
||||||
|
|
||||||
var fileMgr = {};
|
var fileMgr = {};
|
||||||
|
@ -117,11 +117,11 @@ define([
|
|||||||
var headers = {
|
var headers = {
|
||||||
'Content-Type': 'multipart/mixed; boundary="' + boundary + '"',
|
'Content-Type': 'multipart/mixed; boundary="' + boundary + '"',
|
||||||
};
|
};
|
||||||
if(etag !== undefined) {
|
// Sometimes we have error 412 from Google even with the correct
|
||||||
// Sometimes we have error 412 from Google even with the correct
|
// etag
|
||||||
// etag
|
// if(etag !== undefined) {
|
||||||
// headers["If-Match"] = etag;
|
// headers["If-Match"] = etag;
|
||||||
}
|
// }
|
||||||
|
|
||||||
var base64Data = utils.encodeBase64(content);
|
var base64Data = utils.encodeBase64(content);
|
||||||
var multipartRequestBody = delimiter + 'Content-Type: application/json\r\n\r\n' + JSON.stringify(metadata) + delimiter + 'Content-Type: ' + contentType + '\r\n' + 'Content-Transfer-Encoding: base64\r\n' + '\r\n' + base64Data + close_delim;
|
var multipartRequestBody = delimiter + 'Content-Type: application/json\r\n\r\n' + JSON.stringify(metadata) + delimiter + 'Content-Type: ' + contentType + '\r\n' + 'Content-Transfer-Encoding: base64\r\n' + '\r\n' + base64Data + close_delim;
|
||||||
@ -402,8 +402,8 @@ define([
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
googleHelper.picker = function(callback) {
|
googleHelper.picker = function(callback, isImagePicker) {
|
||||||
var ids = [];
|
var docs = [];
|
||||||
var picker = undefined;
|
var picker = undefined;
|
||||||
function hidePicker() {
|
function hidePicker() {
|
||||||
if(picker !== undefined) {
|
if(picker !== undefined) {
|
||||||
@ -415,24 +415,23 @@ define([
|
|||||||
connect(task);
|
connect(task);
|
||||||
loadPicker(task);
|
loadPicker(task);
|
||||||
task.onRun(function() {
|
task.onRun(function() {
|
||||||
var view = new google.picker.View(google.picker.ViewId.DOCS);
|
|
||||||
view.setMimeTypes("text/x-markdown,text/plain,application/octet-stream");
|
|
||||||
var pickerBuilder = new google.picker.PickerBuilder();
|
var pickerBuilder = new google.picker.PickerBuilder();
|
||||||
pickerBuilder.enableFeature(google.picker.Feature.NAV_HIDDEN);
|
|
||||||
pickerBuilder.enableFeature(google.picker.Feature.MULTISELECT_ENABLED);
|
|
||||||
pickerBuilder.setAppId(GOOGLE_DRIVE_APP_ID);
|
pickerBuilder.setAppId(GOOGLE_DRIVE_APP_ID);
|
||||||
var token = gapi.auth.getToken();
|
if(!isImagePicker) {
|
||||||
if(token) {
|
var view = new google.picker.View(google.picker.ViewId.DOCS);
|
||||||
pickerBuilder.setOAuthToken(token.access_token);
|
view.setMimeTypes("text/x-markdown,text/plain,application/octet-stream");
|
||||||
|
pickerBuilder.enableFeature(google.picker.Feature.NAV_HIDDEN);
|
||||||
|
pickerBuilder.enableFeature(google.picker.Feature.MULTISELECT_ENABLED);
|
||||||
|
pickerBuilder.addView(view);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
pickerBuilder.addView(google.picker.ViewId.PHOTOS);
|
||||||
|
pickerBuilder.addView(google.picker.ViewId.PHOTO_UPLOAD);
|
||||||
}
|
}
|
||||||
pickerBuilder.addView(view);
|
|
||||||
pickerBuilder.addView(new google.picker.DocsUploadView());
|
|
||||||
pickerBuilder.setCallback(function(data) {
|
pickerBuilder.setCallback(function(data) {
|
||||||
if(data.action == google.picker.Action.PICKED || data.action == google.picker.Action.CANCEL) {
|
if(data.action == google.picker.Action.PICKED || data.action == google.picker.Action.CANCEL) {
|
||||||
if(data.action == google.picker.Action.PICKED) {
|
if(data.action == google.picker.Action.PICKED) {
|
||||||
for ( var i = 0; i < data.docs.length; i++) {
|
docs = data.docs;
|
||||||
ids.push(data.docs[i].id);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
hidePicker();
|
hidePicker();
|
||||||
task.chain();
|
task.chain();
|
||||||
@ -446,7 +445,7 @@ define([
|
|||||||
picker.setVisible(true);
|
picker.setVisible(true);
|
||||||
});
|
});
|
||||||
task.onSuccess(function() {
|
task.onSuccess(function() {
|
||||||
callback(undefined, ids);
|
callback(undefined, docs);
|
||||||
});
|
});
|
||||||
task.onError(function(error) {
|
task.onError(function(error) {
|
||||||
hidePicker();
|
hidePicker();
|
@ -118,14 +118,16 @@
|
|||||||
var that = this,
|
var that = this,
|
||||||
panels;
|
panels;
|
||||||
|
|
||||||
this.run = function (previewWrapper) { // benweet
|
// benweet
|
||||||
|
var undoManager;
|
||||||
|
this.run = function (previewWrapper) {
|
||||||
if (panels)
|
if (panels)
|
||||||
return; // already initialized
|
return; // already initialized
|
||||||
|
|
||||||
panels = new PanelCollection(idPostfix);
|
panels = new PanelCollection(idPostfix);
|
||||||
var commandManager = new CommandManager(hooks, getString);
|
var commandManager = new CommandManager(hooks, getString);
|
||||||
var previewManager = new PreviewManager(markdownConverter, panels, function () { hooks.onPreviewRefresh(); }, previewWrapper); // benweet
|
var previewManager = new PreviewManager(markdownConverter, panels, function () { hooks.onPreviewRefresh(); }, previewWrapper); // benweet
|
||||||
var undoManager, uiManager;
|
var uiManager;
|
||||||
|
|
||||||
if (!/\?noundo/.test(doc.location.href)) {
|
if (!/\?noundo/.test(doc.location.href)) {
|
||||||
undoManager = new UndoManager(function () {
|
undoManager = new UndoManager(function () {
|
||||||
@ -148,6 +150,12 @@
|
|||||||
forceRefresh();
|
forceRefresh();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// benweet
|
||||||
|
this.restart = function() {
|
||||||
|
undoManager.reinit();
|
||||||
|
that.refreshPreview();
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// before: contains all the text in the input box BEFORE the selection.
|
// before: contains all the text in the input box BEFORE the selection.
|
||||||
@ -676,6 +684,18 @@
|
|||||||
saveState();
|
saveState();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// benweet
|
||||||
|
this.reinit = function() {
|
||||||
|
undoStack = [];
|
||||||
|
stackPtr = 0;
|
||||||
|
mode = "none";
|
||||||
|
lastState = undefined;
|
||||||
|
timer = undefined;
|
||||||
|
inputStateObj = undefined;
|
||||||
|
refreshState();
|
||||||
|
saveState();
|
||||||
|
};
|
||||||
|
|
||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
0
js/lib/jquery.js → js/libs/jquery.js
vendored
0
js/lib/jquery.js → js/libs/jquery.js
vendored
142
js/libs/jquery.waitforimages.js
Normal file
142
js/libs/jquery.waitforimages.js
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
/*
|
||||||
|
* waitForImages 1.4.2
|
||||||
|
* -------------------
|
||||||
|
* Provides a callback when all images have loaded in your given selector.
|
||||||
|
* https://github.com/alexanderdickson/waitForImages
|
||||||
|
*
|
||||||
|
* Copyright (c) 2013 Alex Dickson
|
||||||
|
* Licensed under the MIT license.
|
||||||
|
*/
|
||||||
|
(function ($) {
|
||||||
|
// Namespace all events.
|
||||||
|
var eventNamespace = 'waitForImages';
|
||||||
|
|
||||||
|
// CSS properties which contain references to images.
|
||||||
|
$.waitForImages = {
|
||||||
|
hasImageProperties: ['backgroundImage', 'listStyleImage', 'borderImage', 'borderCornerImage']
|
||||||
|
};
|
||||||
|
|
||||||
|
// Custom selector to find `img` elements that have a valid `src` attribute and have not already loaded.
|
||||||
|
$.expr[':'].uncached = function (obj) {
|
||||||
|
// Ensure we are dealing with an `img` element with a valid `src` attribute.
|
||||||
|
if (!$(obj).is('img[src!=""]')) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Firefox's `complete` property will always be `true` even if the image has not been downloaded.
|
||||||
|
// Doing it this way works in Firefox.
|
||||||
|
var img = new Image();
|
||||||
|
img.src = obj.src;
|
||||||
|
return !img.complete;
|
||||||
|
};
|
||||||
|
|
||||||
|
$.fn.waitForImages = function (finishedCallback, eachCallback, waitForAll) {
|
||||||
|
|
||||||
|
var allImgsLength = 0;
|
||||||
|
var allImgsLoaded = 0;
|
||||||
|
|
||||||
|
// Handle options object.
|
||||||
|
if ($.isPlainObject(arguments[0])) {
|
||||||
|
waitForAll = arguments[0].waitForAll;
|
||||||
|
eachCallback = arguments[0].each;
|
||||||
|
// This must be last as arguments[0]
|
||||||
|
// is aliased with finishedCallback.
|
||||||
|
finishedCallback = arguments[0].finished;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle missing callbacks.
|
||||||
|
finishedCallback = finishedCallback || $.noop;
|
||||||
|
eachCallback = eachCallback || $.noop;
|
||||||
|
|
||||||
|
// Convert waitForAll to Boolean
|
||||||
|
waitForAll = !! waitForAll;
|
||||||
|
|
||||||
|
// Ensure callbacks are functions.
|
||||||
|
if (!$.isFunction(finishedCallback) || !$.isFunction(eachCallback)) {
|
||||||
|
throw new TypeError('An invalid callback was supplied.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.each(function () {
|
||||||
|
// Build a list of all imgs, dependent on what images will be considered.
|
||||||
|
var obj = $(this);
|
||||||
|
var allImgs = [];
|
||||||
|
// CSS properties which may contain an image.
|
||||||
|
var hasImgProperties = $.waitForImages.hasImageProperties || [];
|
||||||
|
// To match `url()` references.
|
||||||
|
// Spec: http://www.w3.org/TR/CSS2/syndata.html#value-def-uri
|
||||||
|
var matchUrl = /url\(\s*(['"]?)(.*?)\1\s*\)/g;
|
||||||
|
|
||||||
|
if (waitForAll) {
|
||||||
|
|
||||||
|
// Get all elements (including the original), as any one of them could have a background image.
|
||||||
|
obj.find('*').andSelf().each(function () {
|
||||||
|
var element = $(this);
|
||||||
|
|
||||||
|
// If an `img` element, add it. But keep iterating in case it has a background image too.
|
||||||
|
if (element.is('img:uncached')) {
|
||||||
|
allImgs.push({
|
||||||
|
src: element.attr('src'),
|
||||||
|
element: element[0]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$.each(hasImgProperties, function (i, property) {
|
||||||
|
var propertyValue = element.css(property);
|
||||||
|
var match;
|
||||||
|
|
||||||
|
// If it doesn't contain this property, skip.
|
||||||
|
if (!propertyValue) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get all url() of this element.
|
||||||
|
while (match = matchUrl.exec(propertyValue)) {
|
||||||
|
allImgs.push({
|
||||||
|
src: match[2],
|
||||||
|
element: element[0]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// For images only, the task is simpler.
|
||||||
|
obj.find('img:uncached')
|
||||||
|
.each(function () {
|
||||||
|
allImgs.push({
|
||||||
|
src: this.src,
|
||||||
|
element: this
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
allImgsLength = allImgs.length;
|
||||||
|
allImgsLoaded = 0;
|
||||||
|
|
||||||
|
// If no images found, don't bother.
|
||||||
|
if (allImgsLength === 0) {
|
||||||
|
finishedCallback.call(obj[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$.each(allImgs, function (i, img) {
|
||||||
|
|
||||||
|
var image = new Image();
|
||||||
|
|
||||||
|
// Handle the image loading and error with the same callback.
|
||||||
|
$(image).bind('load.' + eventNamespace + ' error.' + eventNamespace, function (event) {
|
||||||
|
allImgsLoaded++;
|
||||||
|
|
||||||
|
// If an error occurred with loading the image, set the third argument accordingly.
|
||||||
|
eachCallback.call(img.element, allImgsLoaded, allImgsLength, event.type == 'load');
|
||||||
|
|
||||||
|
if (allImgsLoaded == allImgsLength) {
|
||||||
|
finishedCallback.call(obj[0]);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
image.src = img.src;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}(jQuery));
|
101
js/main.js
101
js/main.js
@ -1,63 +1,86 @@
|
|||||||
// RequireJS configuration
|
// RequireJS configuration
|
||||||
requirejs.config({
|
requirejs.config({
|
||||||
waitSeconds: 0,
|
waitSeconds: 0,
|
||||||
paths: {
|
paths: {
|
||||||
"jquery": "lib/jquery",
|
"jquery": "libs/jquery",
|
||||||
"underscore": "lib/underscore",
|
"underscore": "libs/underscore",
|
||||||
"jgrowl": "lib/jgrowl",
|
"jgrowl": "libs/jgrowl",
|
||||||
"lib/MathJax": '../lib/MathJax/MathJax.js?config=TeX-AMS_HTML'
|
"libs/MathJax": '../lib/MathJax/MathJax.js?config=TeX-AMS_HTML'
|
||||||
},
|
},
|
||||||
shim: {
|
shim: {
|
||||||
'underscore': {
|
'underscore': {
|
||||||
exports: '_'
|
exports: '_'
|
||||||
},
|
},
|
||||||
'jgrowl': {
|
'jgrowl': {
|
||||||
deps: ['jquery'],
|
deps: [
|
||||||
|
'jquery'
|
||||||
|
],
|
||||||
exports: 'jQuery.jGrowl'
|
exports: 'jQuery.jGrowl'
|
||||||
},
|
},
|
||||||
'lib/jquery-ui': ['jquery'],
|
'libs/jquery-ui': [
|
||||||
'lib/bootstrap': ['jquery'],
|
'jquery'
|
||||||
'lib/layout': ['lib/jquery-ui'],
|
],
|
||||||
'lib/Markdown.Extra': ['lib/Markdown.Converter', 'lib/prettify'],
|
'libs/bootstrap': [
|
||||||
'lib/Markdown.Editor': ['lib/Markdown.Converter']
|
'jquery'
|
||||||
|
],
|
||||||
|
'libs/jquery.waitforimages': [
|
||||||
|
'jquery'
|
||||||
|
],
|
||||||
|
'libs/layout': [
|
||||||
|
'libs/jquery-ui'
|
||||||
|
],
|
||||||
|
'libs/Markdown.Extra': [
|
||||||
|
'libs/Markdown.Converter',
|
||||||
|
'libs/prettify'
|
||||||
|
],
|
||||||
|
'libs/Markdown.Editor': [
|
||||||
|
'libs/Markdown.Converter'
|
||||||
|
]
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Defines the logger object
|
// Defines the logger object
|
||||||
var logger = {
|
var logger = {
|
||||||
debug: function() {},
|
debug: function() {
|
||||||
log: function() {},
|
},
|
||||||
info: function() {},
|
log: function() {
|
||||||
warn: function() {},
|
},
|
||||||
error: function() {}
|
info: function() {
|
||||||
|
},
|
||||||
|
warn: function() {
|
||||||
|
},
|
||||||
|
error: function() {
|
||||||
|
}
|
||||||
};
|
};
|
||||||
// Use http://.../?console to print logs in the console
|
// We can run StackEdit with http://.../?console to print logs in the console
|
||||||
if (location.search.match(/(\?|&)console/)) {
|
if(location.search.match(/(\?|&)console/)) {
|
||||||
logger = console;
|
logger = console;
|
||||||
}
|
}
|
||||||
|
|
||||||
// RequireJS entry point. By requiring synchronizer and publisher, we are actually loading all the modules
|
// RequireJS entry point. By requiring synchronizer, publisher and
|
||||||
|
// media-importer, we are actually loading all the modules
|
||||||
require([
|
require([
|
||||||
"jquery",
|
"jquery",
|
||||||
"core",
|
"core",
|
||||||
"synchronizer",
|
"synchronizer",
|
||||||
"publisher"
|
"publisher",
|
||||||
|
"media-importer"
|
||||||
], function($, core) {
|
], function($, core) {
|
||||||
|
|
||||||
$(function() {
|
$(function() {
|
||||||
|
|
||||||
// If browser has detected a new application cache.
|
// If browser has detected a new application cache.
|
||||||
if (window.applicationCache) {
|
if(window.applicationCache) {
|
||||||
window.applicationCache.addEventListener('updateready', function(e) {
|
window.applicationCache.addEventListener('updateready', function(e) {
|
||||||
if(window.applicationCache.status === window.applicationCache.UPDATEREADY) {
|
if(window.applicationCache.status === window.applicationCache.UPDATEREADY) {
|
||||||
window.applicationCache.swapCache();
|
window.applicationCache.swapCache();
|
||||||
window.location.reload();
|
window.location.reload();
|
||||||
}
|
}
|
||||||
}, false);
|
}, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Here, all the modules are loaded and the DOM is ready
|
// Here, all the modules are loaded and the DOM is ready
|
||||||
core.setReady();
|
core.setReady();
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
38
js/media-importer.js
Normal file
38
js/media-importer.js
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
define([
|
||||||
|
"jquery",
|
||||||
|
"underscore",
|
||||||
|
"core",
|
||||||
|
"providers/gplus-provider"
|
||||||
|
], function($, _, core) {
|
||||||
|
|
||||||
|
var mediaImporter = {};
|
||||||
|
|
||||||
|
// Create a map with providerId: providerModule
|
||||||
|
var providerMap = _.chain(arguments).map(function(argument) {
|
||||||
|
return argument && argument.providerId && [
|
||||||
|
argument.providerId,
|
||||||
|
argument
|
||||||
|
];
|
||||||
|
}).compact().object().value();
|
||||||
|
|
||||||
|
core.onReady(function() {
|
||||||
|
_.each(providerMap, function(provider) {
|
||||||
|
// Import image action links (if any)
|
||||||
|
$(".action-import-image-" + provider.providerId).click(function() {
|
||||||
|
// Take the insertLinkCallback from core module
|
||||||
|
var insertLinkCallback = core.insertLinkCallback;
|
||||||
|
// Unset it to be sure core module will not call it
|
||||||
|
core.insertLinkCallback = undefined;
|
||||||
|
provider.importImage(function(error, imageLink) {
|
||||||
|
if(error) {
|
||||||
|
insertLinkCallback(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
insertLinkCallback(imageLink || null);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return mediaImporter;
|
||||||
|
});
|
@ -1,7 +1,7 @@
|
|||||||
define([
|
define([
|
||||||
"underscore",
|
"underscore",
|
||||||
"utils",
|
"utils",
|
||||||
"google-helper"
|
"helpers/google-helper"
|
||||||
], function(_, utils, googleHelper) {
|
], function(_, utils, googleHelper) {
|
||||||
|
|
||||||
var PROVIDER_BLOGGER = "blogger";
|
var PROVIDER_BLOGGER = "blogger";
|
@ -3,7 +3,7 @@ define([
|
|||||||
"utils",
|
"utils",
|
||||||
"extension-manager",
|
"extension-manager",
|
||||||
"file-manager",
|
"file-manager",
|
||||||
"dropbox-helper"
|
"helpers/dropbox-helper"
|
||||||
], function(_, utils, extensionMgr, fileMgr, dropboxHelper) {
|
], function(_, utils, extensionMgr, fileMgr, dropboxHelper) {
|
||||||
|
|
||||||
var PROVIDER_DROPBOX = "dropbox";
|
var PROVIDER_DROPBOX = "dropbox";
|
@ -5,7 +5,7 @@ define([
|
|||||||
"settings",
|
"settings",
|
||||||
"extension-manager",
|
"extension-manager",
|
||||||
"file-manager",
|
"file-manager",
|
||||||
"google-helper"
|
"helpers/google-helper"
|
||||||
], function(_, core, utils, settings, extensionMgr, fileMgr, googleHelper) {
|
], function(_, core, utils, settings, extensionMgr, fileMgr, googleHelper) {
|
||||||
|
|
||||||
var PROVIDER_GDRIVE = "gdrive";
|
var PROVIDER_GDRIVE = "gdrive";
|
||||||
@ -62,19 +62,19 @@ define([
|
|||||||
;
|
;
|
||||||
|
|
||||||
gdriveProvider.importFiles = function() {
|
gdriveProvider.importFiles = function() {
|
||||||
googleHelper.picker(function(error, ids) {
|
googleHelper.picker(function(error, docs) {
|
||||||
if(error || ids.length === 0) {
|
if(error || ids.length === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
var importIds = [];
|
var importIds = [];
|
||||||
_.each(ids, function(id) {
|
_.each(docs, function(doc) {
|
||||||
var syncIndex = createSyncIndex(id);
|
var syncIndex = createSyncIndex(doc.id);
|
||||||
var fileDesc = fileMgr.getFileFromSyncIndex(syncIndex);
|
var fileDesc = fileMgr.getFileFromSyncIndex(syncIndex);
|
||||||
if(fileDesc !== undefined) {
|
if(fileDesc !== undefined) {
|
||||||
extensionMgr.onError('"' + fileDesc.title + '" was already imported.');
|
extensionMgr.onError('"' + fileDesc.title + '" was already imported.');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
importIds.push(id);
|
importIds.push(doc.id);
|
||||||
});
|
});
|
||||||
importFilesFromIds(importIds);
|
importFilesFromIds(importIds);
|
||||||
});
|
});
|
@ -1,6 +1,6 @@
|
|||||||
define([
|
define([
|
||||||
"utils",
|
"utils",
|
||||||
"github-helper"
|
"helpers/github-helper"
|
||||||
], function(utils, githubHelper) {
|
], function(utils, githubHelper) {
|
||||||
|
|
||||||
var PROVIDER_GIST = "gist";
|
var PROVIDER_GIST = "gist";
|
@ -1,7 +1,7 @@
|
|||||||
define([
|
define([
|
||||||
"utils",
|
"utils",
|
||||||
"settings",
|
"settings",
|
||||||
"github-helper"
|
"helpers/github-helper"
|
||||||
], function(utils, settings, githubHelper) {
|
], function(utils, settings, githubHelper) {
|
||||||
|
|
||||||
var PROVIDER_GITHUB = "github";
|
var PROVIDER_GITHUB = "github";
|
79
js/providers/gplus-provider.js
Normal file
79
js/providers/gplus-provider.js
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
define([
|
||||||
|
"underscore",
|
||||||
|
"core",
|
||||||
|
"utils",
|
||||||
|
"extension-manager",
|
||||||
|
"helpers/google-helper"
|
||||||
|
], function(_, core, utils, extensionMgr, googleHelper) {
|
||||||
|
|
||||||
|
var PROVIDER_GPLUS = "gplus";
|
||||||
|
|
||||||
|
var gplusProvider = {
|
||||||
|
providerId: PROVIDER_GPLUS,
|
||||||
|
providerName: "Google+"
|
||||||
|
};
|
||||||
|
|
||||||
|
function getThumbnailUrl(doc, size) {
|
||||||
|
var result = undefined;
|
||||||
|
_.find(doc.thumbnails, function(thumbnail) {
|
||||||
|
var found = false;
|
||||||
|
thumbnail.url.replace(/(.*\/s)\d.*?(\/[^\/]+)/, function(match, sub1, sub2) {
|
||||||
|
result = sub1 + size + sub2;
|
||||||
|
found = true;
|
||||||
|
});
|
||||||
|
return found;
|
||||||
|
});
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
var importImageCallback = undefined;
|
||||||
|
var imageDoc = undefined;
|
||||||
|
gplusProvider.importImage = function(callback) {
|
||||||
|
importImageCallback = callback;
|
||||||
|
googleHelper.picker(function(error, docs) {
|
||||||
|
if(error || docs.length === 0) {
|
||||||
|
callback(error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
imageDoc = docs[0];
|
||||||
|
if(!imageDoc.thumbnails) {
|
||||||
|
extensionMgr.onError("Image " + imageDoc.title + " is not accessible.");
|
||||||
|
callback(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
utils.resetModalInputs();
|
||||||
|
$("#modal-import-image img").prop("src", getThumbnailUrl(imageDoc, 128));
|
||||||
|
utils.setInputValue("#input-import-image-title", imageDoc.name);
|
||||||
|
|
||||||
|
// Load preferences
|
||||||
|
var serializedPreferences = localStorage[PROVIDER_GPLUS + ".importImagePreferences"];
|
||||||
|
if(serializedPreferences) {
|
||||||
|
var importImagePreferences = JSON.parse(serializedPreferences);
|
||||||
|
utils.setInputValue("#input-import-image-size", importImagePreferences.size);
|
||||||
|
}
|
||||||
|
|
||||||
|
$("#modal-import-image").modal();
|
||||||
|
}, true);
|
||||||
|
};
|
||||||
|
|
||||||
|
core.onReady(function() {
|
||||||
|
$(".action-import-image").click(function(e) {
|
||||||
|
var size = utils.getInputIntValue("#input-import-image-size", undefined, 0) || 0;
|
||||||
|
var title = utils.getInputTextValue("#input-import-image-title");
|
||||||
|
var image = getThumbnailUrl(imageDoc, size);
|
||||||
|
if(title) {
|
||||||
|
image += ' \"' + title + '"';
|
||||||
|
}
|
||||||
|
importImageCallback(undefined, image);
|
||||||
|
|
||||||
|
// Store import preferences for next time
|
||||||
|
var importImagePreferences = {};
|
||||||
|
if(size) {
|
||||||
|
importImagePreferences.size = size;
|
||||||
|
}
|
||||||
|
localStorage[PROVIDER_GPLUS + ".importImagePreferences"] = JSON.stringify(importImagePreferences);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return gplusProvider;
|
||||||
|
});
|
@ -1,6 +1,6 @@
|
|||||||
define([
|
define([
|
||||||
"utils",
|
"utils",
|
||||||
"ssh-helper"
|
"helpers/ssh-helper"
|
||||||
], function(utils, sshHelper) {
|
], function(utils, sshHelper) {
|
||||||
|
|
||||||
var PROVIDER_SSH = "ssh";
|
var PROVIDER_SSH = "ssh";
|
@ -1,6 +1,6 @@
|
|||||||
define([
|
define([
|
||||||
"utils",
|
"utils",
|
||||||
"tumblr-helper"
|
"helpers/tumblr-helper"
|
||||||
], function(utils, tumblrHelper) {
|
], function(utils, tumblrHelper) {
|
||||||
|
|
||||||
var PROVIDER_TUMBLR = "tumblr";
|
var PROVIDER_TUMBLR = "tumblr";
|
@ -1,6 +1,6 @@
|
|||||||
define([
|
define([
|
||||||
"utils",
|
"utils",
|
||||||
"wordpress-helper"
|
"helpers/wordpress-helper"
|
||||||
], function(utils, wordpressHelper) {
|
], function(utils, wordpressHelper) {
|
||||||
|
|
||||||
var PROVIDER_WORDPRESS = "wordpress";
|
var PROVIDER_WORDPRESS = "wordpress";
|
@ -8,14 +8,14 @@ define([
|
|||||||
"file-system",
|
"file-system",
|
||||||
"file-manager",
|
"file-manager",
|
||||||
"sharing",
|
"sharing",
|
||||||
"blogger-provider",
|
"providers/blogger-provider",
|
||||||
"dropbox-provider",
|
"providers/dropbox-provider",
|
||||||
"gist-provider",
|
"providers/gist-provider",
|
||||||
"github-provider",
|
"providers/github-provider",
|
||||||
"gdrive-provider",
|
"providers/gdrive-provider",
|
||||||
"ssh-provider",
|
"providers/ssh-provider",
|
||||||
"tumblr-provider",
|
"providers/tumblr-provider",
|
||||||
"wordpress-provider"
|
"providers/wordpress-provider"
|
||||||
], function($, _, core, utils, settings, extensionMgr, fileSystem, fileMgr, sharing) {
|
], function($, _, core, utils, settings, extensionMgr, fileSystem, fileMgr, sharing) {
|
||||||
|
|
||||||
var publisher = {};
|
var publisher = {};
|
||||||
|
@ -6,8 +6,8 @@ define([
|
|||||||
"extension-manager",
|
"extension-manager",
|
||||||
"file-manager",
|
"file-manager",
|
||||||
"async-runner",
|
"async-runner",
|
||||||
"download-provider",
|
"providers/download-provider",
|
||||||
"gist-provider"
|
"providers/gist-provider"
|
||||||
], function($, _, core, utils, extensionMgr, fileMgr, asyncRunner) {
|
], function($, _, core, utils, extensionMgr, fileMgr, asyncRunner) {
|
||||||
|
|
||||||
var sharing = {};
|
var sharing = {};
|
||||||
|
@ -6,8 +6,8 @@ define([
|
|||||||
"extension-manager",
|
"extension-manager",
|
||||||
"file-system",
|
"file-system",
|
||||||
"file-manager",
|
"file-manager",
|
||||||
"dropbox-provider",
|
"providers/dropbox-provider",
|
||||||
"gdrive-provider"
|
"providers/gdrive-provider"
|
||||||
], function($, _, core, utils, extensionMgr, fileSystem, fileMgr) {
|
], function($, _, core, utils, extensionMgr, fileSystem, fileMgr) {
|
||||||
|
|
||||||
var synchronizer = {};
|
var synchronizer = {};
|
||||||
@ -142,7 +142,6 @@ define([
|
|||||||
providerList = _.values(providerMap);
|
providerList = _.values(providerMap);
|
||||||
providerDown(callback);
|
providerDown(callback);
|
||||||
}
|
}
|
||||||
;
|
|
||||||
|
|
||||||
// Main entry point for synchronization
|
// Main entry point for synchronization
|
||||||
var syncRunning = false;
|
var syncRunning = false;
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
define([
|
define([
|
||||||
"jquery",
|
"jquery",
|
||||||
"underscore",
|
"underscore",
|
||||||
"lib/FileSaver"
|
"libs/FileSaver"
|
||||||
], function($, _) {
|
], function($, _) {
|
||||||
|
|
||||||
var utils = {};
|
var utils = {};
|
||||||
|
@ -24,7 +24,7 @@
|
|||||||
var require = { baseUrl : "js", deps : [ "main" + suffix ] };
|
var require = { baseUrl : "js", deps : [ "main" + suffix ] };
|
||||||
var viewerMode = true;
|
var viewerMode = true;
|
||||||
</script>
|
</script>
|
||||||
<script src="js/lib/require.js"></script>
|
<script src="js/libs/require.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body class="viewer">
|
<body class="viewer">
|
||||||
<div id="navbar" class="navbar navbar-fixed-top ui-layout-north">
|
<div id="navbar" class="navbar navbar-fixed-top ui-layout-north">
|
||||||
|
Loading…
Reference in New Issue
Block a user