/* eslint no-underscore-dangle: ["error", { "allowAfterThis": true }]*/ /* eslint no-use-before-define: ["error", { "functions": false }]*/ /* eslint-disable no-underscore-dangle */ /* global Uint8Array fabric tui animateSpinner Assets ActiveStoragePreviews PerfectScrollbar MarvinJsEditor refreshProtocolStatusBar */ var FilePreviewModal = (function() { 'use strict'; var readOnly = false; function initPreviewModal(options = {}) { var name; var url; var downloadUrl; readOnly = options.readOnly; $('.file-preview-link').off('click'); $('.file-preview-link').click(function(e) { e.preventDefault(); name = $(this).find('.attachment-label').text(); url = $(this).data('preview-url'); downloadUrl = $(this).attr('href'); openPreviewModal(name, url, downloadUrl); return true; }); $('#filePreviewModal').find('.preview-close').click(function() { $('#filePreviewModal').find('.file-preview-container').html(''); $('#filePreviewModal').modal('hide'); if (typeof refreshProtocolStatusBar === 'function') refreshProtocolStatusBar(); }); } // Adding rotation icon function updateFabricControls() { fabric.Object.prototype.drawBorders = function(ctx, styleOverride = {}) { var wh = this._calculateCurrentDimensions(); var strokeWidth = 1 / this.borderScaleFactor; var width = wh.x + strokeWidth; var height = wh.y + strokeWidth; var drawRotatingPoint = typeof styleOverride.hasRotatingPoint !== 'undefined' ? styleOverride.hasRotatingPoint : this.hasRotatingPoint; var hasControls = typeof styleOverride.hasControls !== 'undefined' ? styleOverride.hasControls : this.hasControls; var rotatingPointOffset = typeof styleOverride.rotatingPointOffset !== 'undefined' ? styleOverride.rotatingPointOffset : this.rotatingPointOffset; var rotateHeight = -height / 2; ctx.save(); ctx.strokeStyle = styleOverride.borderColor || this.borderColor; this._setLineDash(ctx, styleOverride.borderDashArray || this.borderDashArray, null); ctx.strokeRect( -width / 2, -height / 2, width, height ); if (drawRotatingPoint && this.isControlVisible('mtr') && hasControls) { ctx.beginPath(); ctx.moveTo(0, rotateHeight); ctx.lineTo(0, rotateHeight - rotatingPointOffset + 10); ctx.stroke(); } ctx.restore(); return this; }; fabric.Object.prototype.drawControls = function(ctx) { var rotationImage = ''; var rotate = new Image(); var rotateLeft; var rotateTop; var wh = this._calculateCurrentDimensions(); var width = wh.x; var height = wh.y; var scaleOffset = this.cornerSize; var left = -(width + scaleOffset) / 2; var top = -(height + scaleOffset) / 2; var methodName = this.transparentCorners ? 'stroke' : 'fill'; if (!this.hasControls) { return this; } ctx.save(); ctx.strokeStyle = this.cornerColor; ctx.fillStyle = this.cornerColor; if (!this.transparentCorners) { ctx.strokeStyle = this.cornerStrokeColor; } this._setLineDash(ctx, this.cornerDashArray, null); // top-left this._drawControl( 'tl', ctx, methodName, left, top ); // top-right this._drawControl( 'tr', ctx, methodName, left + width, top ); // bottom-left this._drawControl( 'bl', ctx, methodName, left, top + height ); // bottom-right this._drawControl( 'br', ctx, methodName, left + width, top + height ); if (!this.get('lockUniScaling')) { // middle-top this._drawControl( 'mt', ctx, methodName, left + width / 2, top ); // middle-bottom this._drawControl( 'mb', ctx, methodName, left + width / 2, top + height ); // middle-right this._drawControl( 'mr', ctx, methodName, left + width, top + height / 2 ); // middle-left this._drawControl( 'ml', ctx, methodName, left, top + height / 2 ); } // middle-top-rotate if (this.hasRotatingPoint) { rotate.src = rotationImage; rotateLeft = left + width / 2 - 6; rotateTop = top - this.rotatingPointOffset - 6; ctx.drawImage(rotate, rotateLeft, rotateTop, 32, 32); } ctx.restore(); return this; }; } function preInitImageEditor(data) { $.ajax({ url: data['download-url'], type: 'get', success: function(responseData) { var fileUrl = responseData; initImageEditor(data, fileUrl); } }); } function initImageEditor(data, fileUrl) { var imageEditor; var ps; var blackTheme = { 'common.bi.image': '', 'common.bisize.width': '0', 'common.bisize.height': '0', 'common.backgroundImage': 'none', 'common.backgroundColor': '#1e1e1e', 'common.border': '0px', // header 'header.backgroundImage': 'none', 'header.backgroundColor': 'transparent', 'header.border': '0px', // load button 'loadButton.backgroundColor': '#fff', 'loadButton.border': '1px solid #ddd', 'loadButton.color': '#222', 'loadButton.fontFamily': '\'Noto Sans\', sans-serif', 'loadButton.fontSize': '12px', // download button 'downloadButton.backgroundColor': '#fdba3b', 'downloadButton.border': '1px solid #fdba3b', 'downloadButton.color': '#fff', 'downloadButton.fontFamily': '\'Noto Sans\', sans-serif', 'downloadButton.fontSize': '12px', // main icons 'menu.normalIcon.path': '/images/icon-d.svg', 'menu.normalIcon.name': 'icon-d', 'menu.activeIcon.path': '/images/icon-b.svg', 'menu.activeIcon.name': 'icon-b', 'menu.disabledIcon.path': '/images/icon-a.svg', 'menu.disabledIcon.name': 'icon-a', 'menu.hoverIcon.path': '/images/icon-c.svg', 'menu.hoverIcon.name': 'icon-c', 'menu.iconSize.width': '24px', 'menu.iconSize.height': '24px', // submenu primary color 'submenu.backgroundColor': '#1e1e1e', 'submenu.partition.color': '#3c3c3c', // submenu icons 'submenu.normalIcon.path': '/images/icon-d.svg', 'submenu.normalIcon.name': 'icon-d', 'submenu.activeIcon.path': '/images/icon-c.svg', 'submenu.activeIcon.name': 'icon-c', 'submenu.iconSize.width': '32px', 'submenu.iconSize.height': '32px', // submenu labels 'submenu.normalLabel.color': '#8a8a8a', 'submenu.normalLabel.fontWeight': 'lighter', 'submenu.activeLabel.color': '#fff', 'submenu.activeLabel.fontWeight': 'lighter', // checkbox style 'checkbox.border': '0px', 'checkbox.backgroundColor': '#fff', // range style 'range.pointer.color': '#fff', 'range.bar.color': '#666', 'range.subbar.color': '#d1d1d1', 'range.disabledPointer.color': '#414141', 'range.disabledBar.color': '#282828', 'range.disabledSubbar.color': '#414141', 'range.value.color': '#fff', 'range.value.fontWeight': 'lighter', 'range.value.fontSize': '11px', 'range.value.border': '1px solid #353535', 'range.value.backgroundColor': '#151515', 'range.title.color': '#fff', 'range.title.fontWeight': 'lighter', // colorpicker style 'colorpicker.button.border': '1px solid #1e1e1e', 'colorpicker.title.color': '#fff' }; animateSpinner(null, true); imageEditor = new tui.ImageEditor('#tui-image-editor', { includeUI: { loadImage: { path: fileUrl, name: data.filename }, theme: blackTheme, initMenu: 'draw', menuBarPosition: 'bottom' }, cssMaxWidth: 700, cssMaxHeight: 500, selectionStyle: { cornerSize: 20, rotatingPointOffset: 70, borderColor: '#333', cornerColor: '#333', cornerStyle: 'circle', borderScaleFactor: 3 }, usageStatistics: false }); imageEditor.on('image_loaded', () => { $('.file-save-link').css('display', ''); animateSpinner(null, false); }); ps = new PerfectScrollbar($('.tui-image-editor-wrap')[0], { wheelSpeed: 0.5 }); $('#tui-image-editor .tui-image-editor').on('mousewheel', (e) => { var imageOriginalSize = { width: imageEditor._graphics.canvasImage.width, height: imageEditor._graphics.canvasImage.height }; var wDelta = e.originalEvent.wheelDelta || e.originalEvent.deltaY; var imageEditorWindow = e.currentTarget; var scrollContainer = $('.tui-image-editor-wrap'); var initWidth = imageEditorWindow.style.width; var initHeight = imageEditorWindow.style.height; var scrollContainerInitial = { top: scrollContainer.scrollTop(), left: scrollContainer.scrollLeft(), height: scrollContainer[0].scrollHeight, width: scrollContainer[0].scrollWidth }; var mousePosition = { top: e.clientY - (imageEditorWindow.offsetTop - scrollContainerInitial.top), left: e.clientX - $(imageEditorWindow).offset().left }; var newWidth; var newHeight; var offsetY; var offsetX; if (wDelta > 0) { newWidth = parseInt(initWidth, 10) * 1.1; newHeight = parseInt(initHeight, 10) * 1.1; if (newWidth > imageOriginalSize.width || newHeight > imageOriginalSize.height) { newWidth = imageOriginalSize.width; newHeight = imageOriginalSize.height; } } else { newWidth = parseInt(initWidth, 10) * 0.9; newHeight = parseInt(initHeight, 10) * 0.9; if (parseInt(imageEditorWindow.dataset.minWidth, 10) * 0.5 > parseInt(newWidth, 10)) { newWidth = parseInt(imageEditorWindow.dataset.minWidth, 10) * 0.5; newHeight = parseInt(imageEditorWindow.dataset.minHeight, 10) * 0.5; } } imageEditorWindow.style.width = newWidth + 'px'; imageEditorWindow.style.height = newHeight + 'px'; $(imageEditorWindow).find('canvas, .tui-image-editor-canvas-container') .css('max-width', imageEditorWindow.style.width) .css('max-height', imageEditorWindow.style.height); if (imageEditorWindow.dataset.minHeight === undefined) { imageEditorWindow.dataset.minHeight = initHeight; imageEditorWindow.dataset.minWidth = initWidth; } offsetY = (scrollContainer[0].scrollHeight - scrollContainerInitial.height) * (mousePosition.top / scrollContainerInitial.height); offsetX = (scrollContainer[0].scrollWidth - scrollContainerInitial.width) * (mousePosition.left / scrollContainerInitial.width); scrollContainer.scrollTop(scrollContainerInitial.top + offsetY); scrollContainer.scrollLeft(scrollContainerInitial.left + offsetX); ps.update(); e.preventDefault(); e.stopPropagation(); }); $('.tui-image-editor-wrap')[0].onwheel = function() { return false; }; $('.tui-image-editor-wrap').css('height', 'calc(100% - 150px)'); $('#fileEditModal').find('.file-name').text('Editing: ' + data.filename); $('#fileEditModal').modal('show'); $('.tui-image-editor-header').hide(); $('.file-save-link').css('display', 'none'); $('.file-save-link').off().click(function(ev) { var imageBlob; var imageDataURL; var imageParams; var dataUpload = new FormData(); var blobArray; var bytePosition; ev.preventDefault(); ev.stopPropagation(); if (data['mime-type'] === 'image/png') { imageParams = { format: 'png' }; } else { imageParams = { format: 'jpeg', quality: (data.quality / 100) }; } imageDataURL = imageEditor.toDataURL(imageParams); imageDataURL = atob(imageDataURL.split(',')[1]); blobArray = new Uint8Array(imageDataURL.length); for (bytePosition = 0; bytePosition < imageDataURL.length; bytePosition += 1) { blobArray[bytePosition] = imageDataURL.charCodeAt(bytePosition); } imageBlob = new Blob([blobArray]); dataUpload.append('image', imageBlob); animateSpinner(null, true); $.ajax({ type: 'POST', url: '/files/' + data.id + '/update_image', data: dataUpload, contentType: false, processData: false, success: function(res) { $('#modal_link' + data.id).parent().html(res.html); } }).done(function() { animateSpinner(null, false); imageEditor.destroy(); imageEditor = {}; $('#tui-image-editor').html(''); $('#fileEditModal').modal('hide'); }); if (data.mode === 'tinymce') { $.ajax({ type: 'PUT', url: data.url, data: dataUpload, contentType: false, processData: false, success: function(res) { data.image.src = res.url; } }).done(function() { closeEditor(); }); } else { $.ajax({ type: 'POST', url: '/files/' + data.id + '/update_image', data: dataUpload, contentType: false, processData: false, success: function(res) { $('#modal_link' + data.id).parent().html(res.html); Assets.setupAssetsLoading(); } }).done(function() { closeEditor(); }); } if (typeof refreshProtocolStatusBar === 'function') refreshProtocolStatusBar(); }); window.onresize = function() { imageEditor.ui.resizeEditor(); }; } function openPreviewModal(name, url, downloadUrl) { var modal = $('#filePreviewModal'); updateFabricControls(); $.ajax({ url: url, type: 'GET', dataType: 'json', success: function(data) { var link = modal.find('.file-download-link'); clearPrevieModal(); if (Object.prototype.hasOwnProperty.call(data, 'wopi-controls')) { modal.find('.file-wopi-controls').html(data['wopi-controls']); } link.attr('href', downloadUrl); link.attr('data-no-turbolink', true); link.attr('data-status', 'asset-present'); if (data.type === 'image') { animateSpinner('.file-preview-container', false); modal.find('.file-preview-container') .append($('') .css('opacity', 0) .attr('src', data['large-preview-url']) .attr('alt', name) .on('error', ActiveStoragePreviews.reCheckPreview) .on('load', ActiveStoragePreviews.showPreview) .click(function(ev) { ev.stopPropagation(); })); if (!readOnly && data.editable) { modal.find('.file-edit-link').css('display', ''); modal.find('.file-edit-link').off().click(function(ev) { $.post('/files/' + data.id + '/start_edit_image'); ev.preventDefault(); ev.stopPropagation(); modal.modal('hide'); preInitImageEditor(data); }); } else { modal.find('.file-edit-link').css('display', 'none'); } } else if (data.type === 'marvinjs') { openMarvinEditModal(data, modal); } else { modal.find('.file-edit-link').css('display', 'none'); modal.find('.file-preview-container').html(data['preview-icon']); } if (readOnly) { modal.find('#wopi_file_edit_button').remove(); } modal.find('.file-name').text(name); modal.modal(); modal.find('a[disabled=disabled]').click(function(ev) { ev.preventDefault(); }); $('.modal-backdrop').last().css('z-index', modal.css('z-index') - 1); }, error: function() { // TODO } }); } function clearPrevieModal() { var modal = $('#filePreviewModal'); modal.find('.file-preview-container').empty(); modal.find('.file-wopi-controls').empty(); modal.find('.file-edit-link').css('display', 'none'); } function openMarvinEditModal(data, modal) { modal.find('.file-preview-container') .append($('') .css('opacity', 0) .attr('src', data['large-preview-url']) .attr('alt', data.name) .on('error', ActiveStoragePreviews.reCheckPreview) .on('load', ActiveStoragePreviews.showPreview) .click(function(ev) { ev.stopPropagation(); })); if (!readOnly && data.editable) { modal.find('.file-edit-link').css('display', ''); modal.find('.file-edit-link').off().click(function(ev) { ev.preventDefault(); ev.stopPropagation(); modal.modal('hide'); $.post(data['update-url'] + '/start_editing'); MarvinJsEditor.open({ mode: 'edit', data: data.description, name: data.name, marvinUrl: data['update-url'] }); }); } else { modal.find('.file-edit-link').css('display', 'none'); } } return Object.freeze({ init: initPreviewModal, imageEditor: initImageEditor }); }(window));