From 723024202e08267c61b703654b200a1b9b3f549a Mon Sep 17 00:00:00 2001
From: aignatov-bio <47317017+aignatov-bio@users.noreply.github.com>
Date: Fri, 22 Mar 2019 09:27:50 +0100
Subject: [PATCH] Image editor improvements [SCI - 3158, 3160, 3156, 3154]
(#1563)
* Improvments SCI - 3158, 3160, 3156, 3154
---
.../javascripts/sitewide/file_preview.js | 413 ++++++++++++++++++
.../javascripts/sitewide/file_preview.js.erb | 262 -----------
package.json | 6 +-
3 files changed, 416 insertions(+), 265 deletions(-)
create mode 100644 app/assets/javascripts/sitewide/file_preview.js
delete mode 100644 app/assets/javascripts/sitewide/file_preview.js.erb
diff --git a/app/assets/javascripts/sitewide/file_preview.js b/app/assets/javascripts/sitewide/file_preview.js
new file mode 100644
index 000000000..2305061eb
--- /dev/null
+++ b/app/assets/javascripts/sitewide/file_preview.js
@@ -0,0 +1,413 @@
+/* eslint no-underscore-dangle: ["error", { "allowAfterThis": true }]*/
+/* eslint no-use-before-define: ["error", { "functions": false }]*/
+/* global fabric tui animateSpinner setupAssetsLoading I18n*/
+//= require assets
+
+(function(global) {
+ 'use strict';
+
+ global.initPreviewModal = function initPreviewModal() {
+ var name;
+ var url;
+ var downloadUrl;
+ $('.file-preview-link').off('click');
+ $('.file-preview-link').click(function(e) {
+ e.preventDefault();
+ name = $(this).find('p').text();
+ url = $(this).data('preview-url');
+ downloadUrl = $(this).attr('href');
+ openPreviewModal(name, url, downloadUrl);
+ });
+ };
+
+ // 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 initImageEditor(data) {
+ var imageEditor;
+ 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'
+ };
+
+ imageEditor = new tui.ImageEditor('#tui-image-editor', {
+ includeUI: {
+ loadImage: {
+ path: data['download-url'],
+ 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
+ });
+
+ $('#fileEditModal').find('.file-name').text('Editing: ' + data.filename);
+ $('#fileEditModal').modal('show');
+
+ $('.tui-image-editor-header').hide();
+
+ $('.file-save-link').off().click(function(ev) {
+ var dataUpload = new FormData();
+ ev.preventDefault();
+ ev.stopPropagation();
+ dataUpload.append('image', imageEditor.toDataURL());
+ 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);
+ setupAssetsLoading();
+ }
+ }).done(function() {
+ animateSpinner(null, false);
+ imageEditor.destroy();
+ imageEditor = {};
+ $('#tui-image-editor').html('');
+ $('#fileEditModal').modal('hide');
+ });
+ });
+
+ 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');
+ modal.find('.file-preview-container').empty();
+ modal.find('.file-wopi-controls').empty();
+ 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') {
+ if (data.processing) {
+ animateSpinner('.file-preview-container', true);
+ } else {
+ animateSpinner('.file-preview-container', false);
+ modal.find('.file-preview-container')
+ .append($('')
+ .attr('src', data['large-preview-url'])
+ .attr('alt', name)
+ .click(function(ev) {
+ ev.stopPropagation();
+ }));
+ if (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');
+ initImageEditor(data);
+ });
+ } else {
+ modal.find('.file-edit-link').css('display', 'none');
+ }
+ }
+ } else {
+ modal.find('.file-preview-container').html(data['preview-icon']);
+ }
+ if (data.processing) {
+ checkFileReady(url, modal);
+ }
+ modal.find('.file-name').text(name);
+ modal.find('.preview-close').click(function() {
+ modal.modal('hide');
+ });
+ 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 checkFileReady(url, modal) {
+ $.get(url, function(data) {
+ if (data.processing) {
+ $('.file-download-link')
+ .addClass('disabled-with-click-events')
+ .attr(
+ 'title',
+ I18n.t('general.file.processing')
+ )
+ .click(function(ev) {
+ ev.preventDefault();
+ ev.stopPropagation();
+ });
+ setTimeout(function() {
+ checkFileReady(url, modal);
+ }, 10000);
+ } else {
+ if (data.type === 'image') {
+ modal.find('.file-preview-container').empty();
+ modal.find('.file-preview-container')
+ .append($('')
+ .attr('src', data['large-preview-url'])
+ .attr('alt', data.filename)
+ .click(function(ev) {
+ ev.stopPropagation();
+ }));
+ modal.find('.file-name').text(data.filename);
+ modal.find('.modal-body').click(function() {
+ modal.modal('hide');
+ });
+ modal.modal();
+ $('.modal-backdrop').last().css('z-index', modal.css('z-index') - 1);
+ }
+ $('.file-download-link')
+ .removeClass('disabled-with-click-events')
+ .removeAttr('title')
+ .off();
+ }
+ });
+ }
+}(window));
diff --git a/app/assets/javascripts/sitewide/file_preview.js.erb b/app/assets/javascripts/sitewide/file_preview.js.erb
deleted file mode 100644
index f6dcbafb5..000000000
--- a/app/assets/javascripts/sitewide/file_preview.js.erb
+++ /dev/null
@@ -1,262 +0,0 @@
-
-//= require assets
-
-(function(global) {
- 'use strict';
-
- global.initPreviewModal = function initPreviewModal() {
- var name;
- var url;
- var downloadUrl;
- $('.file-preview-link').off('click');
- $('.file-preview-link').click(function(e) {
- e.preventDefault();
- name = $(this).find('p').text();
- url = $(this).data('preview-url');
- downloadUrl = $(this).attr('href');
- openPreviewModal(name, url, downloadUrl);
- });
- }
-
- function initImageEditor(data) {
- var imageEditor;
- 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'
- };
-
- imageEditor = new tui.ImageEditor('#tui-image-editor', {
- includeUI: {
- loadImage: {
- path: data['download-url'],
- name: data.filename
- },
- theme: blackTheme, // or whiteTheme
- menu: ['draw', 'text', 'shape', 'crop', 'flip', 'icon', 'filter'],
- initMenu: 'draw',
- menuBarPosition: 'bottom'
- },
- cssMaxWidth: 700,
- cssMaxHeight: 500,
- selectionStyle: {
- cornerSize: 20,
- rotatingPointOffset: 70
- },
- usageStatistics: false
- });
-
- $('#fileEditModal').find('.file-name').text('Editing: ' + data.filename);
- $('#fileEditModal').modal('show');
-
- $('.tui-image-editor-header').hide();
-
- $('.file-save-link').off().click(function(ev) {
- var dataUpload = new FormData();
- ev.preventDefault();
- ev.stopPropagation();
- dataUpload.append('image', imageEditor.toDataURL());
- 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);
- setupAssetsLoading();
- }
- }).done(function() {
- animateSpinner(null, false);
- imageEditor.destroy();
- imageEditor = {};
- $('#tui-image-editor').html('');
- $('#fileEditModal').modal('hide');
- });
- });
-
- window.onresize = function() {
- imageEditor.ui.resizeEditor();
- };
- }
-
- function openPreviewModal(name, url, downloadUrl) {
- var modal = $('#filePreviewModal');
- $.ajax({
- url: url,
- type: 'GET',
- dataType: 'json',
- success: function(data) {
- modal.find('.file-preview-container').empty();
- modal.find('.file-wopi-controls').empty();
- if (data.hasOwnProperty('wopi-controls')) {
- modal.find('.file-wopi-controls').html(data['wopi-controls']);
- }
- var link = modal.find('.file-download-link');
- link.attr('href', downloadUrl);
- link.attr('data-no-turbolink', true);
- link.attr('data-status', 'asset-present');
- if (data['type'] === 'image') {
- if(data['processing']) {
- animateSpinner('.file-preview-container', true);
- } else {
- animateSpinner('.file-preview-container', false);
- modal.find('.file-preview-container')
- .append($('')
- .attr('src', data['large-preview-url'])
- .attr('alt', name)
- .click(function(ev) {
- ev.stopPropagation();
- }));
- if (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');
- initImageEditor(data);
- });
- }else{
- modal.find('.file-edit-link').css('display','none');
- }
- }
- } else {
- modal.find('.file-preview-container').html(data['preview-icon']);
- }
- if(data['processing']) {
- checkFileReady(url, modal);
- }
- modal.find('.file-name').text(name);
- modal.find('.preview-close').click(function() {
- modal.modal('hide');
- });
- 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(ev) {
- // TODO
- }
- });
- }
-
- function checkFileReady(url, modal) {
- $.get(url, function(data) {
- if(data['processing']) {
- $('.file-download-link')
- .addClass('disabled-with-click-events')
- .attr('title',
- '<%= I18n.t('general.file.processing')%>')
- .click(function(ev) {
- ev.preventDefault();
- ev.stopPropagation();
- });
- setTimeout(function() {
- checkFileReady(url, modal);
- }, 10000);
- } else {
- if(data['type'] === 'image') {
- modal.find('.file-preview-container').empty();
- modal.find('.file-preview-container')
- .append($('')
- .attr('src', data['large-preview-url'])
- .attr('alt', data['filename'])
- .click(function(ev) {
- ev.stopPropagation();
- })
- );
- modal.find('.file-name').text(data['filename']);
- modal.find('.modal-body').click(function() {
- modal.modal('hide');
- });
- modal.modal();
- $('.modal-backdrop').last().css('z-index', modal.css('z-index') - 1);
- }
- $('.file-download-link')
- .removeClass('disabled-with-click-events')
- .removeAttr('title')
- .off();
- }
- })
- }
-})(window);
diff --git a/package.json b/package.json
index 3fada9c7b..6f6838c6a 100644
--- a/package.json
+++ b/package.json
@@ -60,7 +60,7 @@
"compression-webpack-plugin": "^1.1.11",
"css-loader": "^0.28.11",
"extract-text-webpack-plugin": "^3.0.2",
- "fabric": "^1.6.7",
+ "fabric": "1.6.7",
"file-loader": "^0.11.2",
"glob": "^7.1.2",
"immutability-helper": "^2.7.1",
@@ -101,9 +101,9 @@
"shortid": "^2.2.12",
"style-loader": "^0.18.2",
"styled-components": "^2.4.1",
- "tui-code-snippet": "^1.4.0",
+ "tui-code-snippet": "^1.5.0",
"tui-color-picker": "^2.2.0",
- "tui-image-editor": "^3.2.2",
+ "tui-image-editor": "git://github.com/biosistemika/tui.image-editor",
"webpack": "^3.12.0",
"webpack-manifest-plugin": "^1.3.2",
"webpack-merge": "^4.1.3"