define [ "base" "ide/pdfng/directives/pdfTextLayer" "ide/pdfng/directives/pdfAnnotations" "ide/pdfng/directives/pdfHighlights" "ide/pdfng/directives/pdfRenderer" "ide/pdfng/directives/pdfPage" "ide/pdfng/directives/pdfSpinner" ], ( App pdfTextLayer pdfAnnotations pdfHighlights pdfRenderer pdfPage pdfSpinner pdf ) -> # App = angular.module 'pdfViewerApp', ['pdfPage', 'PDFRenderer', 'pdfHighlights'] App.controller 'pdfViewerController', ['$scope', '$q', '$timeout', 'PDFRenderer', '$element', 'pdfHighlights', 'pdfSpinner', ($scope, $q, $timeout, PDFRenderer, $element, pdfHighlights, pdfSpinner) -> @load = () -> # $scope.pages = [] $scope.document.destroy() if $scope.document? $scope.loadCount = if $scope.loadCount? then $scope.loadCount + 1 else 1 # TODO need a proper url manipulation library to add to query string url = $scope.pdfSrc # add 'pdfng=true' to show that we are using the angular pdfjs viewer queryStringExists = /\?/.test(url) url = url + (if not queryStringExists then '?' else '&') + 'pdfng=true' # for isolated compiles, load the pdf on-demand because nobody will overwrite it onDemandLoading = true $scope.document = new PDFRenderer(url, { scale: 1, disableAutoFetch: if onDemandLoading then true else undefined navigateFn: (ref) -> # this function captures clicks on the annotation links $scope.navigateTo = ref $scope.$apply() progressCallback: (progress) -> $scope.$emit 'progress', progress loadedCallback: () -> $scope.$emit 'loaded' errorCallback: (error) -> Raven?.captureMessage?('pdfng error ' + error) $scope.$emit 'pdf:error', error pageSizeChangeCallback: (pageNum, deltaH) -> $scope.$broadcast 'pdf:page:size-change', pageNum, deltaH }) # we will have all the main information needed to start display # after the following promise is resolved $scope.loaded = $q.all({ numPages: $scope.document.getNumPages() # get size of first page as default @ scale 1 pdfViewport: $scope.document.getPdfViewport 1, 1 }).then (result) -> $scope.pdfViewport = result.pdfViewport $scope.pdfPageSize = [ result.pdfViewport.height, result.pdfViewport.width ] # console.log 'resolved q.all, page size is', result $scope.$emit 'loaded' $scope.numPages = result.numPages .catch (error) -> $scope.$emit 'pdf:error', error return $q.reject(error) @setScale = (scale, containerHeight, containerWidth) -> $scope.loaded.then () -> scale = {} if not scale? if containerHeight == 0 or containerWidth == 0 numScale = 1 else if scale.scaleMode == 'scale_mode_fit_width' # TODO make this dynamic numScale = (containerWidth - 40) / ($scope.pdfPageSize[1]) else if scale.scaleMode == 'scale_mode_fit_height' # TODO magic numbers for jquery ui layout numScale = (containerHeight - 20) / ($scope.pdfPageSize[0]) else if scale.scaleMode == 'scale_mode_value' numScale = scale.scale else if scale.scaleMode == 'scale_mode_auto' # TODO else scale.scaleMode = 'scale_mode_fit_width' numScale = (containerWidth - 40) / ($scope.pdfPageSize[1]) # TODO $scope.scale.scale = numScale $scope.document.setScale(numScale) $scope.defaultPageSize = [ numScale * $scope.pdfPageSize[0], numScale * $scope.pdfPageSize[1] ] # console.log 'in setScale result', $scope.scale.scale, $scope.defaultPageSize .catch (error) -> $scope.$emit 'pdf:error', error return $q.reject(error) @redraw = (position) -> # console.log 'in redraw' # console.log 'reseting pages array for', $scope.numPages $scope.pages = ({ pageNum: i + 1 } for i in [0 .. $scope.numPages-1]) if position? && position.page? # console.log 'position is', position.page, position.offset # console.log 'setting current page', position.page pagenum = position.page if pagenum > $scope.numPages - 1 pagenum = $scope.numPages - 1 $scope.pages[pagenum].current = true $scope.pages[pagenum].position = position @zoomIn = () -> # console.log 'zoom in' newScale = $scope.scale.scale * 1.2 $scope.forceScale = { scaleMode: 'scale_mode_value', scale: newScale } @zoomOut = () -> # console.log 'zoom out' newScale = $scope.scale.scale / 1.2 $scope.forceScale = { scaleMode: 'scale_mode_value', scale: newScale } @fitWidth = () -> # console.log 'fit width' $scope.forceScale = { scaleMode: 'scale_mode_fit_width' } @fitHeight = () -> # console.log 'fit height' $scope.forceScale = { scaleMode: 'scale_mode_fit_height' } @checkPosition = () -> # console.log 'check position' $scope.forceCheck = ($scope.forceCheck || 0) + 1 @showRandomHighlights = () -> # console.log 'show highlights' $scope.highlights = [ { page: 3 h: 100 v: 100 height: 30 width: 200 } ] # we work with (pagenumber, % of height down page from top) # pdfListView works with (pagenumber, vertical position up page from # bottom measured in pts) @getPdfPosition = () -> # console.log 'in getPdfPosition' topPageIdx = 0 topPage = $scope.pages[0] # find first visible page visible = $scope.pages.some (page, i) -> [topPageIdx, topPage] = [i, page] if page.visible if visible && topPage.element? # console.log 'found it', topPageIdx else # console.log 'CANNOT FIND TOP PAGE' return # console.log 'top page is', topPage.pageNum, topPage.elemTop, topPage.elemBottom, topPage top = topPage.element.offset().top bottom = top + topPage.element.innerHeight() viewportTop = $element.offset().top viewportBottom = viewportTop + $element.height() topVisible = (top >= viewportTop && top < viewportBottom) someContentVisible = (top < viewportTop && bottom > viewportTop) # console.log 'in PdfListView', top, topVisible, someContentVisible, viewportTop if topVisible canvasOffset = 0 else if someContentVisible canvasOffset = viewportTop - top else canvasOffset = null # console.log 'pdfListview position = ', canvasOffset # instead of using promise, check if size is known and revert to # default otherwise # console.log 'looking up viewport', topPage.viewport, $scope.pdfViewport if topPage.viewport viewport = topPage.viewport pdfOffset = viewport.convertToPdfPoint(0, canvasOffset); else # console.log 'WARNING: had to default to global page size' viewport = $scope.pdfViewport scaledOffset = canvasOffset / $scope.scale.scale pdfOffset = viewport.convertToPdfPoint(0, scaledOffset); # console.log 'converted to offset = ', pdfOffset newPosition = { "page": topPageIdx, "offset" : { "top" : pdfOffset[1], "left": 0} "pageSize": { "height": viewport.viewBox[3], "width": viewport.viewBox[2] } } return newPosition @computeOffset = (page, position) -> # console.log 'computing offset for', page, position element = page.element #console.log 'element =', $(element), 'parent =', $(element).parent() t1 = $(element).offset()?.top t2 = $(element).parent().offset()?.top if not (t1? and t2?) return $q((resolve, reject) -> reject('elements destroyed')) pageTop = $(element).offset().top - $(element).parent().offset().top # console.log('top of page scroll is', pageTop, 'vs', page.elemTop) # console.log('inner height is', $(element).innerHeight()) currentScroll = $(element).parent().scrollTop() offset = position.offset # convert offset to pixels return $scope.document.getPdfViewport(page.pageNum).then (viewport) -> page.viewport = viewport pageOffset = viewport.convertToViewportPoint(offset.left, offset.top) # if the passed-in position doesn't have the page height/width add them now position.pageSize ?= {"height": viewport.viewBox[3], "width": viewport.viewBox[2]} # console.log 'addition offset =', pageOffset # console.log 'total', pageTop + pageOffset[1] Math.round(pageTop + pageOffset[1] + currentScroll) ## 10 is margin @setPdfPosition = (page, position) -> # console.log 'required pdf Position is', position @computeOffset(page, position).then (offset) -> $scope.pleaseScrollTo = offset $scope.position = position return this ] App.directive 'pdfViewer', ['$q', '$timeout', 'pdfSpinner', ($q, $timeout, pdfSpinner) -> { controller: 'pdfViewerController' controllerAs: 'ctrl' scope: { "pdfSrc": "=" "highlights": "=" "position": "=" "scale": "=" "pleaseJumpTo": "=" } template: """
""" link: (scope, element, attrs, ctrl) -> # console.log 'in pdfViewer element is', element # console.log 'attrs', attrs spinner = new pdfSpinner layoutReady = $q.defer() layoutReady.notify 'waiting for layout' layoutReady.promise.then () -> # console.log 'layoutReady was resolved' renderVisiblePages = () -> visiblePages = getVisiblePages() pages = getExtraPages visiblePages scope.document.renderPages(pages) getVisiblePages = () -> top = element[0].scrollTop; bottom = top + element[0].clientHeight; visiblePages = _.filter scope.pages, (page) -> return false if not page.element? pageElement = page.element[0] pageTop = pageElement.offsetTop pageBottom = pageTop + pageElement.clientHeight page.visible = pageTop < bottom and pageBottom > top return page.visible return visiblePages getExtraPages = (visiblePages) -> extra = [] if visiblePages.length > 0 firstVisiblePage = visiblePages[0].pageNum firstVisiblePageIdx = firstVisiblePage - 1 len = visiblePages.length lastVisiblePage = visiblePages[len-1].pageNum lastVisiblePageIdx = lastVisiblePage - 1 # first page after if lastVisiblePageIdx + 1 < scope.pages.length extra.push scope.pages[lastVisiblePageIdx + 1] # page before if firstVisiblePageIdx > 0 extra.push scope.pages[firstVisiblePageIdx - 1] # second page after if lastVisiblePageIdx + 2 < scope.pages.length extra.push scope.pages[lastVisiblePageIdx + 2] return visiblePages.concat extra rescaleTimer = null queueRescale = (scale) -> # console.log 'call to queueRescale' return if rescaleTimer? or layoutTimer? or elementTimer? # console.log 'adding to rescale queue' rescaleTimer = setTimeout () -> doRescale scale rescaleTimer = null , 0 spinnerTimer = null doRescale = (scale) -> # console.log 'doRescale', scale return unless scale? origposition = angular.copy scope.position # console.log 'origposition', origposition if not spinnerTimer? spinnerTimer = setTimeout () -> spinner.add(element) spinnerTimer = null , 100 layoutReady.promise.then (parentSize) -> [h, w] = parentSize # console.log 'in promise', h, w ctrl.setScale(scale, h, w).then () -> # console.log 'in setscale then', scale, h, w scope.$evalAsync () -> if spinnerTimer clearTimeout spinnerTimer else spinner.remove(element) # stop displaying the text layer element.removeClass 'pdfjs-viewer-show-text' ctrl.redraw(origposition) $timeout renderVisiblePages scope.loadSuccess = true .catch (error) -> scope.$emit 'pdf:error', error elementTimer = null updateLayout = () -> # if element is zero-sized keep checking until it is ready # console.log 'checking element ready', element.height(), element.width() if element.height() == 0 or element.width() == 0 return if elementTimer? elementTimer = setTimeout () -> elementTimer = null updateLayout() , 1000 else scope.parentSize = [ element.innerHeight(), element.innerWidth() ] # console.log 'resolving layoutReady with', scope.parentSize $timeout () -> layoutReady.resolve scope.parentSize scope.$emit 'flash-controls' layoutTimer = null queueLayout = () -> # console.log 'call to queue layout' return if layoutTimer? # console.log 'added to queue layoyt' layoutReady = $q.defer() layoutTimer = setTimeout () -> # console.log 'calling update layout' updateLayout() # console.log 'setting layout timer to null' layoutTimer = null , 0 queueLayout() #scope.$on 'layout:pdf:view', (e, args) -> # console.log 'pdf view change', element, e, args # queueLayout() scope.$on 'layout:main:resize', () -> # console.log 'GOT LAYOUT-MAIN-RESIZE EVENT' queueLayout() scope.$on 'layout:pdf:resize', () -> # FIXME we get this event twice # also we need to start a new layout when we get it # console.log 'GOT LAYOUT-PDF-RESIZE EVENT' queueLayout() scope.$on 'pdf:error', (event, error) -> return if error == 'cancelled' # check if too many retries or file is missing message = error?.message or error if scope.loadCount > 3 || /^Missing PDF/i.test(message) || /^loading/i.test(message) scope.$emit 'pdf:error:display' return if scope.loadSuccess ctrl.load().then () -> # trigger a redraw scope.scale = angular.copy (scope.scale) .catch (error) -> scope.$emit 'pdf:error:display' else scope.$emit 'pdf:error:display' return scope.$on 'pdf:page:size-change', (event, pageNum, delta) -> #console.log 'page size change event', pageNum, delta origposition = angular.copy scope.position #console.log 'orig position', JSON.stringify(origposition) if origposition? && pageNum - 1 < origposition.page && delta != 0 currentScrollTop = element.scrollTop() #console.log 'adjusting scroll from', currentScrollTop, 'by', delta scope.adjustingScroll = true element.scrollTop(currentScrollTop + delta) element.on 'mousedown', (e) -> # We're checking that the event target isn't the directive root element # to make sure that the click was within a PDF page - no point in showing # the text layer when the click is outside. # If the user clicks a PDF page, the mousedown target will be the canvas # element (or the text layer one). Alternatively, if the event target is # the root element, we can assume that the user has clicked either the # grey background area or the scrollbars. if e.target != element[0] and !_hasSelection() element.addClass 'pdfjs-viewer-show-text' _setMouseUpHandler() mouseUpHandler = null # keep track of the handler to avoid adding multiple times _setMouseUpHandler = () -> if not mouseUpHandler? mouseUpHandler = $(document.body).one 'mouseup', _handleSelectionMouseUp _handleSelectionMouseUp = () -> mouseUpHandler = null # reset handler, has now fired window.setTimeout () -> removedClass = _removeClassIfNoSelection() # if we still have a selection we need to keep the handler going if not removedClass then _setMouseUpHandler() , 10 return true _removeClassIfNoSelection = () -> if _hasSelection() return false # didn't remove the text layer else element.removeClass 'pdfjs-viewer-show-text' return true _hasSelection = () -> selection = window.getSelection?() # check the selection collapsed state in preference to # using selection.toString() as the latter is "" when # the selection is hidden (e.g. while viewing logs) return selection? and _isSelectionWithinPDF(selection) and !selection.isCollapsed _isSelectionWithinPDF = (selection) -> if selection.rangeCount == 0 return false selectionAncestorNode = selection.getRangeAt(0).commonAncestorContainer return element.find(selectionAncestorNode).length > 0 or element.is(selectionAncestorNode) element.on 'scroll', () -> #console.log 'scroll event', element.scrollTop(), 'adjusting?', scope.adjustingScroll #scope.scrollPosition = element.scrollTop() if scope.adjustingScroll renderVisiblePages() scope.adjustingScroll = false return if scope.scrollHandlerTimeout clearTimeout(scope.scrollHandlerTimeout) scope.scrollHandlerTimeout = setTimeout scrollHandler, 25 scrollHandler = () -> renderVisiblePages() newPosition = ctrl.getPdfPosition() if newPosition? scope.position = newPosition scope.scrollHandlerTimeout = null scope.$watch 'pdfSrc', (newVal, oldVal) -> # console.log 'loading pdf', newVal, oldVal return unless newVal? scope.loadCount = 0; # new pdf, so reset load count scope.loadSuccess = false ctrl.load().then () -> # trigger a redraw scope.scale = angular.copy (scope.scale) .catch (error) -> scope.$emit 'pdf:error', error scope.$watch 'scale', (newVal, oldVal) -> # no need to set scale when initialising, done in pdfSrc return if newVal == oldVal # console.log 'XXX calling Setscale in scale watch' queueRescale newVal scope.$watch 'forceScale', (newVal, oldVal) -> # console.log 'got change in numscale watcher', newVal, oldVal return unless newVal? queueRescale newVal # scope.$watch 'position', (newVal, oldVal) -> # console.log 'got change in position watcher', newVal, oldVal scope.$watch 'forceCheck', (newVal, oldVal) -> # console.log 'forceCheck', newVal, oldVal return unless newVal? scope.adjustingScroll = true # temporarily disable scroll queueRescale scope.scale scope.$watch('parentSize', (newVal, oldVal) -> # console.log 'XXX in parentSize watch', newVal, oldVal # if newVal == oldVal # console.log 'returning because old and new are the same' # return # return unless oldVal? # console.log 'XXX calling setScale in parentSize watcher' return unless newVal? queueRescale scope.scale , true) # scope.$watch 'elementWidth', (newVal, oldVal) -> # console.log '*** watch INTERVAL element width is', newVal, oldVal scope.$watch 'pleaseScrollTo', (newVal, oldVal) -> # console.log 'got request to ScrollTo', newVal, 'oldVal', oldVal return unless newVal? scope.adjustingScroll = true # temporarily disable scroll # handler while we reposition $(element).scrollTop(newVal) scope.pleaseScrollTo = undefined scope.$watch 'pleaseJumpTo', (newPosition, oldPosition) -> # console.log 'in pleaseJumpTo', newPosition, oldPosition return unless newPosition? ctrl.setPdfPosition scope.pages[newPosition.page-1], newPosition scope.$watch 'navigateTo', (newVal, oldVal) -> return unless newVal? # console.log 'got request to navigate to', newVal, 'oldVal', oldVal scope.navigateTo = undefined # console.log 'navigate to', newVal # console.log 'look up page num' scope.document.getDestination(newVal.dest).then (r) -> # console.log 'need to go to', r # console.log 'page ref is', r[0] scope.document.getPageIndex(r[0]).then (pidx) -> # console.log 'page num is', pidx page = scope.pages[pidx] scope.document.getPdfViewport(page.pageNum).then (viewport) -> #console.log 'got viewport', viewport coords = viewport.convertToViewportPoint r[2], r[3] #console.log 'viewport position', coords #console.log 'r is', r, 'r[1]', r[1], 'r[1].name', r[1].name if r[1].name == 'XYZ' #console.log 'XYZ:', r[2], r[3] newPosition = {page: pidx, offset: {top: r[3], left: r[2]}} ctrl.setPdfPosition scope.pages[pidx], newPosition # XXX? scope.$watch "highlights", (areas) -> # console.log 'got HIGHLIGHTS in pdfViewer', areas return if !areas? #console.log 'areas are', areas highlights = for area in areas or [] { page: area.page - 1 highlight: left: area.h top: area.v height: area.height width: area.width } #console.log 'highlights', highlights return if !highlights.length scope.$broadcast 'pdf:highlights', areas first = highlights[0] # switching between split and full pdf views can cause # highlights to appear before rendering if !scope.pages return # ignore highlight scroll if still rendering pageNum = scope.pages[first.page]?.pageNum if !pageNum? return # ignore highlight scroll if page not found # use a visual offset of 72pt to match the offset in PdfController syncToCode scope.document.getPdfViewport(pageNum).then (viewport) -> position = { page: first.page offset: left: first.highlight.left top: viewport.viewBox[3] - first.highlight.top + first.highlight.height + 72 } ctrl.setPdfPosition(scope.pages[first.page], position) scope.$on '$destroy', () -> # console.log 'handle pdfng directive destroy' clearTimeout elementTimer if elementTimer? clearTimeout layoutTimer if layoutTimer? clearTimeout rescaleTimer if rescaleTimer? clearTimeout spinnerTimer if spinnerTimer? } ]