define [ "track-changes/models/ChangeList" "track-changes/models/Diff" "track-changes/ChangeListView" "track-changes/DiffView" "utils/Modal" "models/Doc" "moment" ], (ChangeList, Diff, ChangeListView, DiffView, Modal, Doc, moment) -> class TrackChangesManager template: $("#trackChangesPanelTemplate").html() constructor: (@ide) -> @project_id = window.userSettings.project_id @$el = $(@template) @ide.mainAreaManager.addArea identifier: "trackChanges" element: @$el @ide.tabManager.addTab id: "history" name: "History" show: "code" after: "code" contract: true onShown: () => @show() onHidden: () => @hide() @ide.editor.on "resize", () => @diffView?.resize() @$el.find(".track-changes-close").on "click", (e) => e.preventDefault @hide() @bindToFileTreeEvents() @disable() bindToFileTreeEvents: () -> @ide.fileTreeManager.on "open:doc", (doc_id) => if @enabled @doc_id = doc_id @updateDiff() show: () -> @changes = new ChangeList([], project_id: @project_id, ide: @ide) @changeListView = new ChangeListView( collection : @changes, el : @$el.find(".change-list-area") ) @changeListView.render() @changeListView.loadUntilFull (error) => @autoSelectDiff() @changeListView.on "change_diff", (fromIndex, toIndex) => @selectDocAndUpdateDiff(fromIndex, toIndex) if @diffView? @diffView.remove() @ide.mainAreaManager.change "trackChanges" @ide.editor.disable() @ide.fileViewManager.disable() @enable() hide: () -> @ide.editor.enable() @ide.fileViewManager.enable() @disable() @ide.fileTreeManager.openDoc(@doc_id) @ide.tabManager.show "code" autoSelectDiff: () -> if @changes.models.length == 0 return # Find all change until the last one we made fromIndex = null for change, i in @changes.models if ide.user in change.get("users") if i > 0 fromIndex = i - 1 else fromIndex = 0 break fromIndex = 0 if !fromIndex toChange = @changes.models[0] fromChange = @changes.models[fromIndex] @changeListView.setSelectionRange(fromIndex, 0) @updateDiff() selectDocAndUpdateDiff: (fromIndex, toIndex) -> doc_ids = [] for change in @changes.models.slice(toIndex, fromIndex + 1) for doc in change.get("docs") or [] doc_ids.push doc.id if doc.id not in doc_ids if !@doc_id? or @doc_id not in doc_ids @doc_id = doc_ids[0] @updateDiff() updateDiff: () -> fromIndex = @changeListView.selectedFromIndex toIndex = @changeListView.selectedToIndex if !toIndex? or !fromIndex? console.log "No selection - what should we do!?" return {from, to, start_ts, end_ts} = @_findDocVersionsRangeInSelection(@doc_id, fromIndex, toIndex) @diff = new Diff({ project_id: @project_id doc_id: @doc_id from: from to: to start_ts: start_ts end_ts: end_ts }, { ide: @ide }) if @diffView? @diffView.remove() @diffView = new DiffView( model: @diff el: @$el.find(".track-changes-diff") ) @diffView.on "restore", () => @restoreDiff(@diff) @diff.fetch() @ide.fileTreeManager.selectEntity(@doc_id) _findDocVersionsRangeInSelection: (doc_id, fromIndex, toIndex) -> from = to = start_ts = end_ts = null for change in @changes.models.slice(toIndex, fromIndex + 1) for doc in change.get("docs") if doc.id == doc_id if from? and to? from = Math.min(from, doc.fromV) to = Math.max(to, doc.toV) start_ts = Math.min(start_ts, change.get("start_ts")) end_ts = Math.max(end_ts, change.get("end_ts")) else from = doc.fromV to = doc.toV start_ts = change.get("start_ts") end_ts = change.get("end_ts") break return {from, to, start_ts, end_ts} restoreDiff: (diff) -> name = diff.get("doc")?.get("name") date = moment(diff.get("start_ts")).format("Do MMM YYYY, h:mm:ss a") modal = new Modal({ title: "Restore document" message: "Are you sure you want to restore #{name} to before the changes on #{date}?" buttons: [{ text: "Cancel" }, { text: "Restore" class: "btn-success" close: false callback: ($button) => $button.text("Restoring...") $button.prop("disabled", true) diff.restore (error) => modal.remove() @hide() }] }) enable: () -> @enabled = true disable: () -> @enabled = false return TrackChangesManager