overleaf/services/web/public/coffee/ide/editor/EditorManager.coffee

189 lines
5.6 KiB
CoffeeScript
Raw Normal View History

2014-07-08 11:02:26 +00:00
define [
"ide/editor/Document"
"ide/editor/directives/aceEditor"
"ide/editor/controllers/SavingNotificationController"
], (Document) ->
class EditorManager
constructor: (@ide, @$scope) ->
@$scope.editor = {
sharejs_doc: null
open_doc_id: null
open_doc_name: null
2014-07-08 11:02:26 +00:00
opening: true
trackChanges: false
2017-02-13 11:09:47 +00:00
wantTrackChanges: false
2014-07-08 11:02:26 +00:00
}
@$scope.$on "entity:selected", (event, entity) =>
if (@$scope.ui.view != "history" and entity.type == "doc")
2014-07-08 11:02:26 +00:00
@openDoc(entity)
@$scope.$on "entity:deleted", (event, entity) =>
if @$scope.editor.open_doc_id == entity.id
return if !@$scope.project.rootDoc_id
doc = @ide.fileTreeManager.findEntityById(@$scope.project.rootDoc_id)
return if !doc?
@openDoc(doc)
2014-07-08 11:02:26 +00:00
initialized = false
@$scope.$on "file-tree:initialized", () =>
if !initialized
initialized = true
@autoOpenDoc()
@$scope.$on "flush-changes", () =>
Document.flushAll()
@$scope.$watch "editor.wantTrackChanges", (value) =>
return if !value?
@_syncTrackChangesState(@$scope.editor.sharejs_doc)
2014-07-08 11:02:26 +00:00
autoOpenDoc: () ->
open_doc_id =
@ide.localStorage("doc.open_id.#{@$scope.project_id}") or
2014-07-08 11:02:26 +00:00
@$scope.project.rootDoc_id
return if !open_doc_id?
doc = @ide.fileTreeManager.findEntityById(open_doc_id)
return if !doc?
@openDoc(doc)
openDocId: (doc_id, options = {}) ->
doc = @ide.fileTreeManager.findEntityById(doc_id)
return if !doc?
@openDoc(doc, options)
2014-07-08 11:02:26 +00:00
openDoc: (doc, options = {}) ->
2016-05-27 13:14:08 +00:00
sl_console.log "[openDoc] Opening #{doc.id}"
2014-07-08 11:02:26 +00:00
@$scope.ui.view = "editor"
done = () =>
if options.gotoLine?
2016-08-11 15:46:12 +00:00
# allow Ace to display document before moving, delay until next tick
2016-08-16 09:59:27 +00:00
# added delay to make this happen later that gotoStoredPosition in
# CursorPositionManager
2016-08-11 15:46:12 +00:00
setTimeout () =>
@$scope.$broadcast "editor:gotoLine", options.gotoLine, options.gotoColumn
, 0
else if options.gotoOffset?
setTimeout () =>
@$scope.$broadcast "editor:gotoOffset", options.gotoOffset
, 0
2016-08-11 15:46:12 +00:00
2014-07-08 11:02:26 +00:00
if doc.id == @$scope.editor.open_doc_id and !options.forceReopen
@$scope.$apply () =>
done()
return
@$scope.editor.open_doc_id = doc.id
@$scope.editor.open_doc_name = doc.name
2014-07-08 11:02:26 +00:00
@ide.localStorage "doc.open_id.#{@$scope.project_id}", doc.id
2014-07-08 11:02:26 +00:00
@ide.fileTreeManager.selectEntity(doc)
@$scope.editor.opening = true
@_openNewDocument doc, (error, sharejs_doc) =>
if error?
2014-07-24 15:07:43 +00:00
@ide.showGenericMessageModal(
"Error opening document"
"Sorry, something went wrong opening this document. Please try again."
)
2014-07-08 11:02:26 +00:00
return
@_syncTrackChangesState(sharejs_doc)
2014-07-08 11:02:26 +00:00
@$scope.$broadcast "doc:opened"
@$scope.$apply () =>
@$scope.editor.opening = false
@$scope.editor.sharejs_doc = sharejs_doc
done()
_openNewDocument: (doc, callback = (error, sharejs_doc) ->) ->
2016-05-27 13:14:08 +00:00
sl_console.log "[_openNewDocument] Opening..."
2014-07-08 11:02:26 +00:00
current_sharejs_doc = @$scope.editor.sharejs_doc
if current_sharejs_doc?
2016-05-27 13:14:08 +00:00
sl_console.log "[_openNewDocument] Leaving existing open doc..."
2014-07-08 11:02:26 +00:00
current_sharejs_doc.leaveAndCleanUp()
@_unbindFromDocumentEvents(current_sharejs_doc)
new_sharejs_doc = Document.getDocument @ide, doc.id
new_sharejs_doc.join (error) =>
return callback(error) if error?
@_bindToDocumentEvents(doc, new_sharejs_doc)
callback null, new_sharejs_doc
2014-07-08 11:02:26 +00:00
_bindToDocumentEvents: (doc, sharejs_doc) ->
sharejs_doc.on "error", (error, meta) =>
if error?.message?
message = error.message
else if typeof error == "string"
message = error
else
message = ""
if message.match "maxDocLength"
@ide.showGenericMessageModal(
"Document Too Long"
"Sorry, this file is too long to be edited manually. Please upload it directly."
)
else if message.match "too many comments or tracked changes"
@ide.showGenericMessageModal(
"Too many comments or tracked changes"
"Sorry, this file has too many comments or tracked changes. Please try accepting or rejecting some existing changes, or resolving and deleting some comments."
)
else
@ide.socket.disconnect()
@ide.reportError(error, meta)
@ide.showGenericMessageModal(
"Out of sync"
2016-11-04 10:25:08 +00:00
"Sorry, this file has gone out of sync and we need to do a full refresh. <br> <a href='/learn/Kb/Editor_out_of_sync_problems'>Please see this help guide for more information</a>"
)
2014-07-08 11:02:26 +00:00
@openDoc(doc, forceReopen: true)
sharejs_doc.on "externalUpdate", (update) =>
return if @_ignoreExternalUpdates
2014-07-08 11:02:26 +00:00
@ide.showGenericMessageModal(
"Document Updated Externally"
"This document was just updated externally. Any recent changes you have made may have been overwritten. To see previous versions please look in the history."
)
_unbindFromDocumentEvents: (document) ->
document.off()
getCurrentDocValue: () ->
@$scope.editor.sharejs_doc?.getSnapshot()
getCurrentDocId: () ->
@$scope.editor.open_doc_id
startIgnoringExternalUpdates: () ->
@_ignoreExternalUpdates = true
stopIgnoringExternalUpdates: () ->
@_ignoreExternalUpdates = false
_syncTimeout: null
_syncTrackChangesState: (doc) ->
return if !doc?
if @_syncTimeout?
clearTimeout @_syncTimeout
@_syncTimeout = null
want = @$scope.editor.wantTrackChanges
have = doc.getTrackingChanges()
if want == have
2017-01-27 16:13:11 +00:00
@$scope.editor.trackChanges = want
return
do tryToggle = () =>
saved = !doc.getInflightOp()? and !doc.getPendingOp()?
if saved
doc.setTrackingChanges(want)
@$scope.$apply () =>
@$scope.editor.trackChanges = want
else
@_syncTimeout = setTimeout tryToggle, 100