/*global gapi */ define([ "jquery", "underscore", "constants", "utils", "storage", "logger", "classes/Provider", "settings", "eventMgr", "fileMgr", "helpers/googleHelper", "text!html/dialogExportGdrive.html", "text!html/dialogAutoSyncGdrive.html", ], function($, _, constants, utils, storage, logger, Provider, settings, eventMgr, fileMgr, googleHelper, dialogExportGdriveHTML, dialogAutoSyncGdriveHTML) { return function(providerId, providerName, accountIndex) { var accountId = 'google.gdrive' + accountIndex; var gdriveProvider = new Provider(providerId, providerName); gdriveProvider.defaultPublishFormat = "template"; gdriveProvider.exportPreferencesInputIds = [ providerId + "-parentid", providerId + "-realtime", ]; function createSyncIndex(id) { return "sync." + providerId + "." + id; } function createSyncAttributes(id, etag, content, title) { var syncAttributes = {}; syncAttributes.provider = gdriveProvider; syncAttributes.id = id; syncAttributes.etag = etag; syncAttributes.contentCRC = utils.crc32(content); syncAttributes.titleCRC = utils.crc32(title); syncAttributes.syncIndex = createSyncIndex(id); return syncAttributes; } function importFilesFromIds(ids) { googleHelper.downloadMetadata(ids, accountId, function(error, result) { if(error) { return; } googleHelper.downloadContent(result, accountId, function(error, result) { if(error) { return; } var fileDescList = []; var fileDesc; _.each(result, function(file) { var syncAttributes = createSyncAttributes(file.id, file.etag, file.content, file.title); syncAttributes.isRealtime = file.isRealtime; var syncLocations = {}; syncLocations[syncAttributes.syncIndex] = syncAttributes; var parsedContent = gdriveProvider.parseSerializedContent(file.content); fileDesc = fileMgr.createFile(file.title, parsedContent.content, parsedContent.discussionList, syncLocations); fileDescList.push(fileDesc); }); if(fileDesc !== undefined) { eventMgr.onSyncImportSuccess(fileDescList, gdriveProvider); fileMgr.selectFile(fileDesc); } }); }); } gdriveProvider.importFiles = function() { googleHelper.picker(function(error, docs) { if(error || docs.length === 0) { return; } var importIds = []; _.each(docs, function(doc) { var syncIndex = createSyncIndex(doc.id); var fileDesc = fileMgr.getFileFromSyncIndex(syncIndex); if(fileDesc !== undefined) { eventMgr.onError('"' + fileDesc.title + '" was already imported.'); return; } importIds.push(doc.id); }); importFilesFromIds(importIds); }, 'doc', accountId); }; gdriveProvider.exportFile = function(event, title, content, callback) { var fileId = utils.getInputTextValue('#input-sync-export-' + providerId + '-fileid'); if(fileId) { // Check that file is not synchronized with another an existing // document var syncIndex = createSyncIndex(fileId); var fileDesc = fileMgr.getFileFromSyncIndex(syncIndex); if(fileDesc !== undefined) { eventMgr.onError('File ID is already synchronized with "' + fileDesc.title + '".'); callback(true); return; } } var parentId = utils.getInputTextValue('#input-sync-export-' + providerId + '-parentid'); googleHelper.upload(fileId, parentId, title, content, undefined, undefined, accountId, function(error, result) { if(error) { callback(error); return; } var syncAttributes = createSyncAttributes(result.id, result.etag, content, title); callback(undefined, syncAttributes); }); }; gdriveProvider.exportRealtimeFile = function(event, title, content, callback) { var parentId = utils.getInputTextValue('#input-sync-export-' + providerId + '-parentid'); googleHelper.createRealtimeFile(parentId, title, accountId, function(error, result) { if(error) { callback(error); return; } var syncAttributes = createSyncAttributes(result.id, result.etag, content, title); callback(undefined, syncAttributes); }); }; var merge = settings.conflictMode == 'merge'; gdriveProvider.syncUp = function(content, contentCRC, title, titleCRC, discussionList, discussionListCRC, syncAttributes, callback) { if( (syncAttributes.contentCRC == contentCRC) && // Content CRC hasn't changed (syncAttributes.titleCRC == titleCRC) && // Content CRC hasn't changed (syncAttributes.discussionListCRC == discussionListCRC) // Discussion list CRC hasn't changed ) { return callback(undefined, false); } var uploadedContent = gdriveProvider.serializeContent(content, discussionList); googleHelper.upload(syncAttributes.id, undefined, title, uploadedContent, undefined, syncAttributes.etag, accountId, function(error, result) { if(error) { callback(error, true); return; } syncAttributes.etag = result.etag; if(merge === true) { // Need to store the whole content for merge syncAttributes.content = content; syncAttributes.title = title; syncAttributes.discussionList = discussionList; } syncAttributes.contentCRC = contentCRC; syncAttributes.titleCRC = titleCRC; syncAttributes.discussionListCRC = discussionListCRC; callback(undefined, true); }); }; gdriveProvider.syncUpRealtime = function(content, contentCRC, title, titleCRC, discussionList, discussionListCRC, syncAttributes, callback) { // Skip if title CRC has not changed if(titleCRC == syncAttributes.titleCRC) { callback(undefined, false); return; } googleHelper.rename(syncAttributes.id, title, accountId, function(error, result) { if(error) { callback(error, true); return; } syncAttributes.etag = result.etag; syncAttributes.titleCRC = titleCRC; callback(undefined, true); }); }; gdriveProvider.syncDown = function(callback) { var lastChangeId = parseInt(storage[accountId + ".gdrive.lastChangeId"], 10); googleHelper.checkChanges(lastChangeId, accountId, function(error, changes, newChangeId) { if(error) { callback(error); return; } var interestingChanges = []; _.each(changes, function(change) { var syncIndex = createSyncIndex(change.fileId); var syncAttributes = fileMgr.getSyncAttributes(syncIndex); if(syncAttributes === undefined) { return; } // Store syncAttributes to avoid 2 times searching change.syncAttributes = syncAttributes; // Delete if(change.deleted === true) { interestingChanges.push(change); return; } // Modify if(syncAttributes.etag != change.file.etag) { interestingChanges.push(change); } }); googleHelper.downloadContent(interestingChanges, accountId, function(error, changes) { if(error) { callback(error); return; } function merge() { if(changes.length === 0) { storage[accountId + ".gdrive.lastChangeId"] = newChangeId; return callback(); } var change = changes.pop(); var syncAttributes = change.syncAttributes; var syncIndex = syncAttributes.syncIndex; var fileDesc = fileMgr.getFileFromSyncIndex(syncIndex); // No file corresponding (file may have been deleted locally) if(fileDesc === undefined) { return; } // File deleted if(change.deleted === true) { eventMgr.onError('"' + fileDesc.title + '" has been removed from ' + providerName + '.'); fileDesc.removeSyncLocation(syncAttributes); eventMgr.onSyncRemoved(fileDesc, syncAttributes); if(syncAttributes.isRealtime === true && fileMgr.currentFile === fileDesc) { gdriveProvider.stopRealtimeSync(); } return; } var file = change.file; var parsedContent = gdriveProvider.parseSerializedContent(file.content); var remoteContent = parsedContent.content; var remoteTitle = file.title; var remoteDiscussionList = parsedContent.discussionList; var remoteCRC = gdriveProvider.syncMerge(fileDesc, syncAttributes, remoteContent, remoteTitle, remoteDiscussionList); // Update syncAttributes syncAttributes.etag = file.etag; if(merge === true) { // Need to store the whole content for merge syncAttributes.content = remoteContent; syncAttributes.title = remoteTitle; syncAttributes.discussionList = remoteDiscussionList; } syncAttributes.contentCRC = remoteCRC.contentCRC; syncAttributes.titleCRC = remoteCRC.titleCRC; syncAttributes.discussionListCRC = remoteCRC.discussionListCRC; utils.storeAttributes(syncAttributes); setTimeout(merge, 5); } setTimeout(merge, 5); }); }); }; gdriveProvider.publish = function(publishAttributes, frontMatter, title, content, callback) { var contentType = publishAttributes.format != "markdown" ? 'text/html' : undefined; googleHelper.upload(publishAttributes.id, undefined, publishAttributes.fileName || title, content, contentType, undefined, accountId, function(error, result) { if(error) { callback(error); return; } publishAttributes.id = result.id; callback(); }); }; gdriveProvider.newPublishAttributes = function(event) { var publishAttributes = {}; publishAttributes.id = utils.getInputTextValue('#input-publish-' + providerId + '-fileid'); publishAttributes.fileName = utils.getInputTextValue('#input-publish-' + providerId + '-filename'); if(event.isPropagationStopped()) { return undefined; } return publishAttributes; }; // Keep a link to the Pagedown editor var pagedownEditor; var undoExecute; var redoExecute; var setUndoRedoButtonStates; eventMgr.addListener("onPagedownConfigure", function(pagedownEditorParam) { pagedownEditor = pagedownEditorParam; }); var realtimeContext; eventMgr.addListener('onContentChanged', function(aceEditorParam) { }); // Start realtime synchronization gdriveProvider.startRealtimeSync = function(fileDesc, syncAttributes) { var context = { fileDesc: fileDesc }; realtimeContext = context; googleHelper.loadRealtime(syncAttributes.id, fileDesc.content, accountId, function(err, doc) { if(err || !doc) { return; } // If user just switched to another document or file has just been // reselected if(context !== realtimeContext) { return doc.close(); } logger.log("Starting Google Drive realtime synchronization"); context.document = doc; var model = doc.getModel(); context.model = realtimeString; var realtimeString = model.getRoot().get('content'); context.string = realtimeString; // Saves model content checksum function updateContentState() { syncAttributes.contentCRC = utils.crc32(realtimeString.getText()); utils.storeAttributes(syncAttributes); } var debouncedRefreshPreview = _.debounce(pagedownEditor.refreshPreview, 100); // Listen to insert text events realtimeString.addEventListener(gapi.drive.realtime.EventType.TEXT_INSERTED, function(e) { if(aceEditor !== undefined && (isAceUpToDate === false || e.isLocal === false)) { // Update ACE editor var position = aceEditor.session.doc.indexToPosition(e.index); aceEditor.session.insert(position, e.text); isAceUpToDate = true; } // If modifications come down from a collaborator if(e.isLocal === false) { logger.log("Google Drive realtime document updated from server"); updateContentState(); aceEditor === undefined && debouncedRefreshPreview(); } }); // Listen to delete text events realtimeString.addEventListener(gapi.drive.realtime.EventType.TEXT_DELETED, function(e) { if(aceEditor !== undefined && (isAceUpToDate === false || e.isLocal === false)) { // Update ACE editor var range = (function(posStart, posEnd) { return new Range(posStart.row, posStart.column, posEnd.row, posEnd.column); })(aceEditor.session.doc.indexToPosition(e.index), aceEditor.session.doc.indexToPosition(e.index + e.text.length)); aceEditor.session.remove(range); isAceUpToDate = true; } // If modifications come down from a collaborator if(e.isLocal === false) { logger.log("Google Drive realtime document updated from server"); updateContentState(); aceEditor === undefined && debouncedRefreshPreview(); } }); doc.addEventListener(gapi.drive.realtime.EventType.DOCUMENT_SAVE_STATE_CHANGED, function(e) { // Save success event if(e.isPending === false && e.isSaving === false) { logger.log("Google Drive realtime document successfully saved on server"); updateContentState(); } }); // Try to merge offline modifications var localContent = fileDesc.content; var localContentChanged = syncAttributes.contentCRC != utils.crc32(localContent); var remoteContent = realtimeString.getText(); var remoteContentCRC = utils.crc32(remoteContent); var remoteContentChanged = syncAttributes.contentCRC != remoteContentCRC; var fileContentChanged = localContent != remoteContent; model.beginCompoundOperation('Open and merge'); if(fileContentChanged === true && localContentChanged === true) { if(remoteContentChanged === true) { // Conflict detected fileMgr.createFile(fileDesc.title + " (backup)", localContent); eventMgr.onMessage('Conflict detected on "' + fileDesc.title + '". A backup has been created locally.'); } else { // Add local modifications if no collaborators change realtimeString.setText(localContent); } } // Update content state according to collaborators changes if(remoteContentChanged === true) { logger.log("Google Drive realtime document updated from server"); aceEditor !== undefined && aceEditor.setValue(remoteContent, -1); updateContentState(); aceEditor === undefined && debouncedRefreshPreview(); } var realtimeDiscussionList = model.getRoot().get('discussionList'); context.discussionList = realtimeDiscussionList; if(!realtimeDiscussionList) { realtimeDiscussionList = model.createMap(); model.getRoot().set('discussionList', realtimeDiscussionList); } mergeDiscussionList(context, remoteContentChanged === true); model.endCompoundOperation(); // Save undo/redo buttons default actions undoExecute = pagedownEditor.uiManager.buttons.undo.execute; redoExecute = pagedownEditor.uiManager.buttons.redo.execute; setUndoRedoButtonStates = pagedownEditor.uiManager.setUndoRedoButtonStates; // Set temporary actions for undo/redo buttons pagedownEditor.uiManager.buttons.undo.execute = function() { if(model.canUndo) { // This flag is used to avoid replaying editor's own // modifications (assuming it's synchronous) isAceUpToDate = false; model.undo(); } }; pagedownEditor.uiManager.buttons.redo.execute = function() { if(model.canRedo) { // This flag is used to avoid replaying editor's own // modifications (assuming it's synchronous) isAceUpToDate = false; model.redo(); } }; // Add event handler for model's UndoRedoStateChanged events pagedownEditor.uiManager.setUndoRedoButtonStates = function() { setTimeout(function() { pagedownEditor.uiManager.setButtonState(pagedownEditor.uiManager.buttons.undo, model.canUndo); pagedownEditor.uiManager.setButtonState(pagedownEditor.uiManager.buttons.redo, model.canRedo); }, 50); }; pagedownEditor.uiManager.setUndoRedoButtonStates(); model.addEventListener(gapi.drive.realtime.EventType.UNDO_REDO_STATE_CHANGED, function() { pagedownEditor.uiManager.setUndoRedoButtonStates(); }); }, function(err) { logger.error(err); if(err.type == "token_refresh_required") { googleHelper.refreshGdriveToken(accountId); } else if(err.type == "not_found") { eventMgr.onError('"' + fileDesc.title + '" has been removed from ' + providerName + '.'); fileDesc.removeSyncLocation(syncAttributes); eventMgr.onSyncRemoved(fileDesc, syncAttributes); gdriveProvider.stopRealtimeSync(); } else if(err.isFatal) { eventMgr.onError('An error has forced real time synchronization to stop.'); gdriveProvider.stopRealtimeSync(); } }); }; function mergeDiscussion(localDiscussion, realtimeDiscussion, takeServer) { if(takeServer) { localDiscussion.selectionStart = realtimeDiscussion.get('selectionStart'); localDiscussion.selectionEnd = realtimeDiscussion.get('selectionEnd'); } else { realtimeDiscussion.set('selectionStart', localDiscussion.selectionStart); realtimeDiscussion.set('selectionEnd', localDiscussion.selectionEnd); } function isCommentInDiscussion(comment, commentList) { return commentList.some(function(commentInDiscussion) { return comment.author == commentInDiscussion.author && comment.content == commentInDiscussion.content; }); } var realtimeCommentList = realtimeDiscussion.get('commentList').asArray(); localDiscussion.commentList.forEach(function(comment) { if(!isCommentInDiscussion(comment, realtimeCommentList)) { realtimeDiscussion.get('commentList').push(comment); } }); realtimeCommentList.forEach(function(comment) { if(!isCommentInDiscussion(comment, localDiscussion.commentList)) { localDiscussion.commentList.push(comment); } }); } function createRealtimeDiscussion(context, discussion) { var commentList = context.model.createList(discussion.commentList); var realtimeDiscussion = context.model.createMap({ selectionStart: discussion.selectionStart, selectionEnd: discussion.selectionEnd, commentList: commentList }); context.discussionList.set(discussion.discussionIndex, realtimeDiscussion); } function mergeDiscussionList(context, takeServer) { var localDiscussionList = context.fileDesc.discussionList; _.each(localDiscussionList, function(localDiscussion) { var realtimeDiscussion = context.discussionList.get(localDiscussion.discussionIndex); if(realtimeDiscussion) { mergeDiscussion(localDiscussion, realtimeDiscussion, takeServer); } else { createRealtimeDiscussion(context, localDiscussion); } }); context.discussionList.keys().forEach(function(discussionIndex) { var localDiscussion = localDiscussionList[discussionIndex]; var realtimeDiscussion = context.discussionList.get(discussionIndex); if(localDiscussion) { mergeDiscussion(localDiscussion, realtimeDiscussion, takeServer); } else { var discussion = { discussionIndex: discussionIndex, selectionStart: realtimeDiscussion.get('selectionStart'), selectionEnd: realtimeDiscussion.get('selectionEnd'), commentList: realtimeDiscussion.get('commentList').asArray() }; localDiscussionList[discussionIndex] = discussion; eventMgr.onCommentsChanged(context.fileDesc); } }); context.fileDesc.discussionList = localDiscussionList; // Write in localStorage } eventMgr.addListener('onCommentsChanged', function(fileDesc, discussion) { if(realtimeContext === undefined || realtimeContext.fileDesc !== fileDesc) { return; } if(!realtimeContext.discussionList.has(discussion.discussionIndex)) { createRealtimeDiscussion(realtimeContext, discussion); } }); // Stop realtime synchronization gdriveProvider.stopRealtimeSync = function() { logger.log("Stopping Google Drive realtime synchronization"); if(realtimeContext !== undefined) { realtimeContext.document && realtimeContext.document.close(); realtimeContext = undefined; } if(setUndoRedoButtonStates !== undefined) { // Set back original undo/redo actions pagedownEditor.uiManager.buttons.undo.execute = undoExecute; pagedownEditor.uiManager.buttons.redo.execute = redoExecute; pagedownEditor.uiManager.setUndoRedoButtonStates = setUndoRedoButtonStates; pagedownEditor.uiManager.setUndoRedoButtonStates(); } }; // Initialize the AutoSync dialog fields gdriveProvider.setAutosyncDialogConfig = function() { var config = gdriveProvider.autosyncConfig; utils.setInputChecked('#input-autosync-' + providerId + '-enabled', config.enabled); utils.setInputValue('#input-autosync-' + providerId + '-parentid', config.parentId); }; // Retrieve the AutoSync dialog fields gdriveProvider.getAutosyncDialogConfig = function() { var config = {}; config.enabled = utils.getInputChecked('#input-autosync-' + providerId + '-enabled'); config.parentId = utils.getInputTextValue('#input-autosync-' + providerId + '-parentid'); return config; }; // Perform AutoSync gdriveProvider.autosyncFile = function(title, content, config, callback) { var parentId = config.parentId; googleHelper.upload(undefined, parentId, title, content, undefined, undefined, accountId, function(error, result) { if(error) { callback(error); return; } var syncAttributes = createSyncAttributes(result.id, result.etag, content, title); callback(undefined, syncAttributes); }); }; // Disable publish on optional multi-account gdriveProvider.isPublishEnabled = settings.gdriveMultiAccount > accountIndex; eventMgr.addListener("onReady", function() { // Hide optional multi-account sub-menus $('.submenu-sync-' + providerId).toggle(settings.gdriveMultiAccount > accountIndex); // Create export dialog var modalUploadElt = document.querySelector('.modal-upload-' + providerId); modalUploadElt && (modalUploadElt.innerHTML = _.template(dialogExportGdriveHTML, { providerId: providerId, providerName: providerName })); // Create autosync dialog var modalAutosyncElt = document.querySelector('.modal-autosync-' + providerId); modalAutosyncElt && (modalAutosyncElt.innerHTML = _.template(dialogAutoSyncGdriveHTML, { providerId: providerId, providerName: providerName })); // Choose folder button in export modal $('.action-export-' + providerId + '-choose-folder').click(function() { googleHelper.picker(function(error, docs) { if(error || docs.length === 0) { return; } // Open export dialog $(".modal-upload-" + providerId).modal(); // Set parent ID utils.setInputValue('#input-sync-export-' + providerId + '-parentid', docs[0].id); }, 'folder', accountId); }); // Choose folder button in autosync modal $('.action-autosync-' + providerId + '-choose-folder').click(function() { googleHelper.picker(function(error, docs) { if(error || docs.length === 0) { return; } // Open export dialog $(".modal-autosync-" + providerId).modal(); // Set parent ID utils.setInputValue('#input-autosync-' + providerId + '-parentid', docs[0].id); }, 'folder', accountId); }); // On export, disable file ID input if realtime is checked var $realtimeCheckboxElt = $('#input-sync-export-' + providerId + '-realtime'); var $fileIdInputElt = $('#input-sync-export-' + providerId + '-fileid'); $('#input-sync-export-' + providerId + '-realtime').change(function() { $fileIdInputElt.prop('disabled', $realtimeCheckboxElt.prop('checked')); }); // Skip gdrive action if provider is not enabled in the settings if(accountIndex >= settings.gdriveMultiAccount) { return; } var state = utils.retrieveIgnoreError(providerId + ".state"); var userId = storage[accountId + '.userId']; if(state === undefined || (userId && state.userId != userId)) { return; } storage.removeItem(providerId + ".state"); if(state.action == "create") { googleHelper.upload(undefined, state.folderId, constants.GDRIVE_DEFAULT_FILE_TITLE, settings.defaultContent, undefined, undefined, accountId, function(error, file) { if(error) { return; } var syncAttributes = createSyncAttributes(file.id, file.etag, file.content, file.title); var syncLocations = {}; syncLocations[syncAttributes.syncIndex] = syncAttributes; var fileDesc = fileMgr.createFile(file.title, file.content, undefined, syncLocations); fileMgr.selectFile(fileDesc); eventMgr.onMessage('"' + file.title + '" created successfully on ' + providerName + '.'); }); } else if(state.action == "open") { var importIds = []; _.each(state.ids, function(id) { var syncIndex = createSyncIndex(id); var fileDesc = fileMgr.getFileFromSyncIndex(syncIndex); if(fileDesc !== undefined) { fileDesc !== fileMgr.currentFile && fileMgr.selectFile(fileDesc); } else { importIds.push(id); } }); importFilesFromIds(importIds); } }); return gdriveProvider; }; });