define( [ "jquery", "utils", "extension-manager", "bootstrap", "layout", "Markdown.Editor", "storage", "config", "underscore", "FileSaver", "css_browser_selector" ], function($, utils, extensionManager) { var core = {}; // For convenience core.doNothing = function() {}; // Time shared by others modules function updateCurrentTime() { core.currentTime = new Date().getTime(); } updateCurrentTime(); // Used for periodic tasks var intervalId = undefined; var periodicCallbacks = []; core.addPeriodicCallback = function(callback) { periodicCallbacks.push(callback); }; // Used to detect user activity var userReal = false; var userActive = false; var windowUnique = true; var userLastActivity = 0; function setUserActive() { userReal = true; userActive = true; userLastActivity = core.currentTime; }; function isUserActive() { if(userActive === true && core.currentTime - userLastActivity > USER_IDLE_THRESHOLD) { userActive = false; } return userActive && windowUnique; } // Used to only have 1 window of the application in the same browser var windowId = undefined; core.checkWindowUnique = function() { if(userReal === false || windowUnique === false) { return; } if(windowId === undefined) { windowId = utils.randomString(); localStorage["frontWindowId"] = windowId; } var frontWindowId = localStorage["frontWindowId"]; if(frontWindowId != windowId) { windowUnique = false; if(intervalId !== undefined) { clearInterval(intervalId); } $(".modal").modal("hide"); $('#modal-non-unique').modal({ backdrop: "static", keyboard: false }); } }; // Export data on disk core.saveFile = function(content, filename) { if(saveAs !== undefined) { var blob = new Blob([content], {type: "text/plain;charset=utf-8"}); saveAs(blob, filename); } else { var uriContent = "data:application/octet-stream;base64," + utils.encodeBase64(content); window.open(uriContent, 'file'); } }; // Used by asyncRunner core.showWorkingIndicator = function(show) { if (show === false) { $(".working-indicator").removeClass("show"); $("body").removeClass("working"); } else { $(".working-indicator").addClass("show"); $("body").addClass("working"); } }; // Log a message core.showMessage = function(message) { console.log(message); extensionManager.onMessage(message); }; // Log an error core.showError = function(error) { console.error(error); if(_.isString(error)) { extensionManager.onMessage(error); } else if(_.isObject(error)) { extensionManager.onMessage(error.message); } }; // Offline management core.isOffline = false; var offlineTime = core.currentTime; var offlineListeners = []; core.addOfflineListener = function(callback) { offlineListeners.push(callback); }; core.setOffline = function() { offlineTime = core.currentTime; if(core.isOffline === false) { core.isOffline = true; extensionManager.onOfflineChanged(true); _.each(offlineListeners, function(listener) { listener(); }); } }; core.setOnline = function() { if(core.isOffline === true) { core.isOffline = false; extensionManager.onOfflineChanged(false); _.each(offlineListeners, function(listener) { listener(); }); } }; function checkOnline() { // Try to reconnect if we are offline but we have some network if (core.isOffline === true && navigator.onLine === true && offlineTime + CHECK_ONLINE_PERIOD < core.currentTime) { offlineTime = core.currentTime; // Try to download anything to test the connection $.ajax({ url : "//www.google.com/jsapi", timeout : AJAX_TIMEOUT, dataType : "script" }).done(function() { core.setOnline(); }); } } // Setting management core.settings = { layoutOrientation : "horizontal", lazyRendering : true, editorFontSize : 14, defaultContent: "\n\n\n> Written with [StackEdit](http://benweet.github.io/stackedit/).", commitMsg : "Published by http://benweet.github.io/stackedit", template : ['\n', '\n', '\n', '<%= documentTitle %>\n', '\n', '<%= documentHTML %>\n', ''].join(""), sshProxy : SSH_PROXY_URL }; if (_.has(localStorage, "settings")) { _.extend(core.settings, JSON.parse(localStorage.settings)); } core.loadSettings = function() { // Layout orientation utils.setInputRadio("radio-layout-orientation", core.settings.layoutOrientation); // Theme utils.setInputValue("#input-settings-theme", localStorage.theme); // Lazy rendering utils.setInputChecked("#input-settings-lazy-rendering", core.settings.lazyRendering); // Editor font size utils.setInputValue("#input-settings-editor-font-size", core.settings.editorFontSize); // Default content utils.setInputValue("#textarea-settings-default-content", core.settings.defaultContent); // Commit message utils.setInputValue("#input-settings-publish-commit-msg", core.settings.commitMsg); // Template utils.setInputValue("#textarea-settings-publish-template", core.settings.template); // SSH proxy utils.setInputValue("#input-settings-ssh-proxy", core.settings.sshProxy); // Load extension settings extensionManager.onLoadSettings(); }; core.saveSettings = function(event) { var newSettings = {}; // Layout orientation newSettings.layoutOrientation = utils.getInputRadio("radio-layout-orientation"); // Theme var theme = utils.getInputValue("#input-settings-theme"); // Lazy Rendering newSettings.lazyRendering = utils.getInputChecked("#input-settings-lazy-rendering"); // Editor font size newSettings.editorFontSize = utils.getInputIntValue("#input-settings-editor-font-size", event, 1, 99); // Default content newSettings.defaultContent = utils.getInputValue("#textarea-settings-default-content"); // Commit message newSettings.commitMsg = utils.getInputTextValue("#input-settings-publish-commit-msg", event); // Template newSettings.template = utils.getInputTextValue("#textarea-settings-publish-template", event); // SSH proxy newSettings.sshProxy = utils.checkUrl(utils.getInputTextValue("#input-settings-ssh-proxy", event), true); // Save extension settings newSettings.extensionSettings = {}; extensionManager.onSaveSettings(newSettings.extensionSettings, event); if(!event.isPropagationStopped()) { $.extend(core.settings, newSettings); localStorage.settings = JSON.stringify(newSettings); localStorage.theme = theme; } }; // Create the layout var layout = undefined; core.createLayout = function() { if(viewerMode === true) { return; } var layoutGlobalConfig = { closable : true, resizable : false, slidable : false, livePaneResizing : true, enableCursorHotkey : false, spacing_open : 15, spacing_closed : 15, togglerLength_open : 90, togglerLength_closed : 90, stateManagement__enabled : false, center__minWidth : 200, center__minHeight : 200 }; extensionManager.onLayoutConfigure(layoutGlobalConfig); if (core.settings.layoutOrientation == "horizontal") { $(".ui-layout-south").remove(); $(".ui-layout-east").addClass("well").prop("id", "wmd-preview"); layout = $('body').layout( $.extend(layoutGlobalConfig, { east__resizable : true, east__size : .5, east__minSize : 200 }) ); } else if (core.settings.layoutOrientation == "vertical") { $(".ui-layout-east").remove(); $(".ui-layout-south").addClass("well").prop("id", "wmd-preview"); layout = $('body').layout( $.extend(layoutGlobalConfig, { south__resizable : true, south__size : .5, south__minSize : 200 }) ); } $(".ui-layout-toggler-north").addClass("btn").append( $("").addClass("caret")); $(".ui-layout-toggler-south").addClass("btn").append( $("").addClass("caret")); $(".ui-layout-toggler-east").addClass("btn").append( $("").addClass("caret")); $("#navbar").click(function() { layout.allowOverflow('north'); }); extensionManager.onLayoutCreated(); }; core.layoutRefresh = function() { if(layout !== undefined) { // Use defer to make sure UI has been updated _.defer(layout.resizeAll); } }; // Create the PageDown editor var insertLinkCallback = undefined; core.createEditor = function(onTextChange) { var converter = new Markdown.Converter(); var editor = new Markdown.Editor(converter); // Custom insert link dialog editor.hooks.set("insertLinkDialog", function (callback) { insertLinkCallback = callback; utils.resetModalInputs(); $("#modal-insert-link").modal(); _.defer(function() { $("#input-insert-link").focus(); }); return true; }); // Custom insert image dialog editor.hooks.set("insertImageDialog", function (callback) { insertLinkCallback = callback; utils.resetModalInputs(); $("#modal-insert-image").modal(); _.defer(function() { $("#input-insert-image").focus(); }); return true; }); var firstChange = true; var previewWrapper = function(makePreview) { return function() { if(firstChange !== true) { onTextChange(); } makePreview(); }; }; if(core.settings.lazyRendering === true) { var lastRefresh = 0; previewWrapper = function(makePreview) { //var debouncedMakePreview = _.debounce(makePreview, 500); return function() { if(firstChange === true) { makePreview(); } else { onTextChange(); var currentDate = new Date().getTime(); if(currentDate - lastRefresh > 500) { makePreview(); lastRefresh = currentDate; } //debouncedMakePreview(); } }; }; } extensionManager.onEditorConfigure(editor); editor.hooks.chain("onPreviewRefresh", extensionManager.onAsyncPreview); // Convert email addresses (not managed by pagedown) converter.hooks.chain("postConversion", function(text) { return text.replace(/<(mailto\:)?([^\s>]+@[^\s>]+\.\S+?)>/g, function(match, mailto, email) { return '' + email + ''; }); }); $("#wmd-input, #wmd-preview").scrollTop(0); $("#wmd-button-bar").empty(); editor.run(previewWrapper); firstChange = false; // Hide default buttons $(".wmd-button-row").addClass("btn-group").find("li:not(.wmd-spacer)") .addClass("btn").css("left", 0).find("span").hide(); // Add customized buttons $("#wmd-bold-button").append($("").addClass("icon-bold")); $("#wmd-italic-button").append($("").addClass("icon-italic")); $("#wmd-link-button").append($("").addClass("icon-globe")); $("#wmd-quote-button").append($("").addClass("icon-indent-left")); $("#wmd-code-button").append($("").addClass("icon-code")); $("#wmd-image-button").append($("").addClass("icon-picture")); $("#wmd-olist-button").append($("").addClass("icon-numbered-list")); $("#wmd-ulist-button").append($("").addClass("icon-list")); $("#wmd-heading-button").append($("").addClass("icon-text-height")); $("#wmd-hr-button").append($("").addClass("icon-hr")); $("#wmd-undo-button").append($("").addClass("icon-undo")); $("#wmd-redo-button").append($("").addClass("icon-share-alt")); }; // Create an centered popup window core.popupWindow = function(url, title, w, h) { var left = (screen.width / 2) - (w / 2); var top = (screen.height / 2) - (h / 2); return window .open( url, title, 'toolbar=no, location=no, directories=no, status=no, menubar=no, scrollbars=no, resizable=no, copyhistory=no, width=' + w + ', height=' + h + ', top=' + top + ', left=' + left); }; // Keep a reference to the fileManager core.setFileManager = function(fileManager) { core.fileManager = fileManager; runReadyCallbacks(); }; // onReady event callbacks var readyCallbacks = []; core.onReady = function(callback) { readyCallbacks.push(callback); runReadyCallbacks(); }; var documentLoaded = false; function runReadyCallbacks() { if(documentLoaded === true && core.fileManager !== undefined) { _.each(readyCallbacks, function(callback) { callback(); }); readyCallbacks = []; } } $(function() { documentLoaded = true; runReadyCallbacks(); }); core.onReady(function() { extensionManager.init(core.settings.extensionSettings); }); core.onReady(function() { // Load theme list _.each(THEME_LIST, function(name, value) { $("#input-settings-theme").append($('')); }); // listen to online/offline events $(window).on('offline', core.setOffline); $(window).on('online', core.setOnline); if (navigator.onLine === false) { core.setOffline(); } // Detect user activity $(document).mousemove(setUserActive).keypress(setUserActive); // Avoid dropdown to close when clicking on submenu $(".dropdown-submenu > a").click(function(e) { e.stopPropagation(); }); // Click events on "insert link" and "insert image" dialog buttons $(".action-insert-link").click(function(e) { var value = utils.getInputTextValue($("#input-insert-link"), e); if(value !== undefined) { insertLinkCallback(value); } }); $(".action-insert-image").click(function(e) { var value = utils.getInputTextValue($("#input-insert-image"), e); if(value !== undefined) { insertLinkCallback(value); } }); $(".action-close-insert-link").click(function(e) { insertLinkCallback(null); }); // Settings loading/saving $(".action-load-settings").click(function() { core.loadSettings(); }); $(".action-apply-settings").click(function(e) { core.saveSettings(e); if(!e.isPropagationStopped()) { window.location.reload(); } }); $(".action-default-settings").click(function() { localStorage.removeItem("settings"); localStorage.removeItem("theme"); window.location.reload(); }); $(".action-app-reset").click(function() { localStorage.clear(); window.location.reload(); }); // UI layout $("#menu-bar, .ui-layout-center, .ui-layout-east, .ui-layout-south").removeClass("hide"); core.createLayout(); // Editor's textarea $("#wmd-input, #md-section-helper").css({ // Apply editor font size "font-size": core.settings.editorFontSize + "px", "line-height": Math.round(core.settings.editorFontSize * (20/14)) + "px" }); // Manage tab key $("#wmd-input").keydown(function(e) { if(e.keyCode === 9) { var value = $(this).val(); var start = this.selectionStart; var end = this.selectionEnd; // IE8 does not support selection attributes if(start === undefined || end === undefined) { return; } $(this).val(value.substring(0, start) + "\t" + value.substring(end)); this.selectionStart = this.selectionEnd = start + 1; e.preventDefault(); } }); // Tooltips $(".tooltip-scroll-link").tooltip({ html: true, container: '#modal-settings', placement: 'right', title: ['Scroll Link is a feature that binds together editor and preview scrollbars. ', 'It allows you to keep an eye on the preview while scrolling the editor and vice versa. ', '

', '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 efficient.', ].join("") }); $(".tooltip-lazy-rendering").tooltip({ container: '#modal-settings', placement: 'right', title: 'Disable preview rendering while typing in order to offload CPU. Refresh preview after 500 ms of inactivity.' }); $(".tooltip-default-content").tooltip({ html: true, container: '#modal-settings', placement: 'right', title: 'Thanks for supporting StackEdit by adding a backlink in your documents!' }); $(".tooltip-template").tooltip({ html: true, container: '#modal-settings', placement: 'right', trigger: 'manual', title: ['Available variables:
', '
  • documentTitle: document title
  • ', '
  • documentMarkdown: document in Markdown format
  • ', '
  • documentHTML: document in HTML format
  • ', '
  • publishAttributes: attributes of the publish location (undefined when using "Save")
', 'Examples:
', _.escape('<%= documentTitle %>'), '
', _.escape('
<%- documentHTML %>
'), '
', _.escape('<% if(publishAttributes.provider == "github") print(documentMarkdown); %>'), '

More info', ].join("") }).click(function(e) { $(this).tooltip('show'); e.stopPropagation(); }); $(document).click(function(e) { $(".tooltip-template").tooltip('hide'); }); // Reset inputs $(".action-reset-input").click(function() { utils.resetModalInputs(); }); // Do periodic tasks intervalId = window.setInterval(function() { updateCurrentTime(); core.checkWindowUnique(); if(isUserActive() === true || viewerMode === true) { _.each(periodicCallbacks, function(callback) { callback(); }); checkOnline(); } }, 1000); }); return core; });