define [ "base", "utils/EventEmitter" "ide/colors/ColorManager" "ide/review-panel/ChangesTracker" ], (App, EventEmitter, ColorManager, ChangesTracker) -> App.controller "ReviewPanelController", ($scope, $element, ide, $timeout) -> $reviewPanelEl = $element.find "#review-panel" $scope.SubViews = CUR_FILE : "cur_file" OVERVIEW : "overview" $scope.reviewPanel = entries: {} trackNewChanges: false hasEntries: false subView: $scope.SubViews.CUR_FILE openSubView: $scope.SubViews.CUR_FILE $scope.commentState = adding: false content: "" $scope.reviewPanelEventsBridge = new EventEmitter() scrollbar = {} $scope.reviewPanelEventsBridge.on "aceScrollbarVisibilityChanged", (isVisible, scrollbarWidth) -> scrollbar = {isVisible, scrollbarWidth} updateScrollbar() updateScrollbar = () -> if scrollbar.isVisible and $scope.reviewPanel.subView == $scope.SubViews.CUR_FILE $reviewPanelEl.css "right", "#{ scrollbar.scrollbarWidth }px" else $reviewPanelEl.css "right", "0" $scope.$watch "reviewPanel.subView", (subView) -> return if !subView? updateScrollbar() $scope.$watch "ui.reviewPanelOpen", (open) -> return if !open? if !open # Always show current file when not open, but save current state $scope.reviewPanel.openSubView = $scope.reviewPanel.subView $scope.reviewPanel.subView = $scope.SubViews.CUR_FILE else # Reset back to what we had when previously open $scope.reviewPanel.subView = $scope.reviewPanel.openSubView changesTrackers = {} $scope.$watch "editor.open_doc_id", (open_doc_id) -> return if !open_doc_id? changesTrackers[open_doc_id] ?= new ChangesTracker() $scope.reviewPanel.changesTracker = changesTrackers[open_doc_id] # TODO Just for prototyping purposes; remove afterwards. mockedUserId = '12345abc' $scope.$watch (() -> entries = $scope.reviewPanel.entries[$scope.editor.open_doc_id] or {} Object.keys(entries).length ), (nEntries) -> $scope.reviewPanel.hasEntries = nEntries > 0 and $scope.trackChangesFeatureFlag $scope.$watch "ui.reviewPanelOpen", (reviewPanelOpen) -> return if !reviewPanelOpen? $timeout () -> $scope.$broadcast "review-panel:toggle" $scope.$broadcast "review-panel:layout" $scope.acceptChange = (entry_id) -> $scope.$broadcast "change:accept", entry_id $scope.rejectChange = (entry_id) -> $scope.$broadcast "change:reject", entry_id $scope.startNewComment = () -> # $scope.commentState.adding = true $scope.$broadcast "comment:select_line" $timeout () -> $scope.$broadcast "review-panel:layout" $scope.submitNewComment = (content) -> # $scope.commentState.adding = false $scope.$broadcast "comment:add", content # $scope.commentState.content = "" $timeout () -> $scope.$broadcast "review-panel:layout" $scope.cancelNewComment = (entry) -> # $scope.commentState.adding = false # $scope.commentState.content = "" $timeout () -> $scope.$broadcast "review-panel:layout" $scope.startReply = (entry) -> entry.replying = true $timeout () -> $scope.$broadcast "review-panel:layout" # $scope.handleCommentReplyKeyPress = (ev, entry) -> # if ev.keyCode == 13 and !ev.shiftKey and !ev.ctrlKey and !ev.metaKey # ev.preventDefault() # ev.target.blur() # $scope.submitReply(entry) $scope.submitReply = (entry, entry_id) -> $scope.unresolveComment(entry_id) entry.thread.push { content: entry.replyContent ts: new Date() user_id: window.user_id } entry.replyContent = "" entry.replying = false $timeout () -> $scope.$broadcast "review-panel:layout" # TODO Just for prototyping purposes; remove afterwards window.setTimeout((() -> $scope.$applyAsync(() -> submitMockedReply(entry)) ), 1000 * 2) # TODO Just for prototyping purposes; remove afterwards. submitMockedReply = (entry) -> entry.thread.push { content: 'Lorem ipsum dolor sit amet' ts: new Date() user_id: mockedUserId } entry.replyContent = "" entry.replying = false $timeout () -> $scope.$broadcast "review-panel:layout" $scope.cancelReply = (entry) -> entry.replying = false entry.replyContent = "" $scope.$broadcast "review-panel:layout" $scope.resolveComment = (entry, entry_id) -> entry.showWhenResolved = false entry.focused = false $scope.$broadcast "comment:resolve", entry_id $scope.unresolveComment = (entry_id) -> $scope.$broadcast "comment:unresolve", entry_id $scope.deleteComment = (entry_id) -> $scope.$broadcast "comment:remove", entry_id $scope.showThread = (entry) -> entry.showWhenResolved = true $timeout () -> $scope.$broadcast "review-panel:layout" $scope.hideThread = (entry) -> entry.showWhenResolved = false $timeout () -> $scope.$broadcast "review-panel:layout" $scope.setSubView = (subView) -> $scope.reviewPanel.subView = subView $scope.gotoEntry = (doc_id, entry) -> console.log "Going to entry", entry.docPos ide.editorManager.openDocId(doc_id, { gotoLine: entry.docPos.row + 1, gotoColumn: entry.docPos.column }) DOC_ID_NAMES = {} $scope.getFileName = (doc_id) -> # This is called a lot and is relatively expensive, so cache the result if !DOC_ID_NAMES[doc_id]? entity = ide.fileTreeManager.findEntityById(doc_id) return if !entity? DOC_ID_NAMES[doc_id] = ide.fileTreeManager.getEntityPath(entity) return DOC_ID_NAMES[doc_id] # TODO: Eventually we need to get this from the server, and update it # when we get an id we don't know. This'll do for client side testing refreshUsers = () -> $scope.users = {} # TODO Just for prototyping purposes; remove afterwards. $scope.users[mockedUserId] = { email: "gerald.butler@gmail.com" name: "Gerald Butler" isSelf: false hue: 70 avatar_text: "G" } for member in $scope.project.members.concat($scope.project.owner) if member._id == window.user_id name = "You" isSelf = true else name = "#{member.first_name} #{member.last_name}" isSelf = false $scope.users[member._id] = { email: member.email name: name isSelf: isSelf hue: ColorManager.getHueForUserId(member._id) avatar_text: [member.first_name, member.last_name].filter((n) -> n?).map((n) -> n[0]).join "" } $scope.$watch "project.members", (members) -> return if !members? refreshUsers()