From 6e33baa995590f0a4d923060b7fe1c2182221de7 Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Mon, 23 Sep 2019 22:56:13 +0200 Subject: [PATCH 01/11] Add skeleton restore-draft notification for module description --- app/assets/javascripts/sitewide/tiny_mce.js | 61 ++++++++++++++++++- app/assets/stylesheets/tiny_mce.scss | 9 +++ .../my_modules/_description_form.html.erb | 4 +- 3 files changed, 72 insertions(+), 2 deletions(-) diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index 86a415a74..6cb71b88b 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -16,6 +16,59 @@ var TinyMCE = (function() { }); } + + + // Get LocalStorage auto save path + function getAutoSavePrefix(editor) { + var prefix = editor.getParam('autosave_prefix', 'tinymce-autosave-{path}{query}{hash}-{id}-'); + + prefix = prefix.replace(/\{path\}/g, document.location.pathname); + prefix = prefix.replace(/\{query\}/g, document.location.search); + prefix = prefix.replace(/\{hash\}/g, document.location.hash); + prefix = prefix.replace(/\{id\}/g, editor.id); + + return prefix; + } + + // Handles autosave notification if draft is available in the local storage + function restoreDraftNotification(selector, editor) { + var prefix = getAutoSavePrefix(editor); + var lastDraftTime = parseInt(tinyMCE.util.LocalStorage.getItem(prefix + 'time'), 10); + var lastUpdated = $(selector).data('last-updated'); + + var restoreBtn = $(''); + var cancelBtn = $('
' + + '
'); + + // Check whether we have draft stored + if (editor.plugins.autosave.hasDraft()) { + var notificationBar = $('
'); + + if (lastDraftTime < lastUpdated) { + notificationBar.text('Older version of the text below has been saved in the browser. Do you want to restore it?'); + } else { + notificationBar.text('Newer version of the text below has been saved in the browser. Do you want to restore it?'); + } + + // Add notification bar + $(notificationBar).append(cancelBtn); + $(notificationBar).append(restoreBtn); + $(editor.contentAreaContainer).before(notificationBar); + + $(restoreBtn).click(function() { + editor.plugins.autosave.restoreDraft(); + editor.plugins.autosave.removeDraft(); + notificationBar.remove(); + }); + + $(cancelBtn).click(function() { + notificationBar.remove(); + }); + } + } + // returns a public API for TinyMCE editor return Object.freeze({ init: function(selector, onSaveCallback) { @@ -60,7 +113,8 @@ var TinyMCE = (function() { browser_spellcheck: true, branding: false, fixed_toolbar_container: '#mytoolbar', - autosave_interval: '15s', + autosave_restore_when_empty: false, + autosave_interval: '1s', autosave_retention: '1440m', removed_menuitems: 'newdocument', object_resizing: true, @@ -117,6 +171,7 @@ var TinyMCE = (function() { var editorForm = $(editor.getContainer()).closest('form'); var menuBar = editorForm.find('.mce-menubar.mce-toolbar.mce-first .mce-flow-layout'); var editorToolbar = editorForm.find('.mce-top-part'); + var editorToolbaroffset; $('.tinymce-placeholder').css('height', $(editor.editorContainer).height() + 'px'); @@ -237,6 +292,10 @@ var TinyMCE = (function() { menuBar.find('.tinymce-save-button').remove(); menuBar.find('.tinymce-cancel-button').remove(); }); + + editor.on('init', function(e) { + restoreDraftNotification(selector, editor); + }); }, codesample_content_css: $(selector).data('highlightjs-path') }); diff --git a/app/assets/stylesheets/tiny_mce.scss b/app/assets/stylesheets/tiny_mce.scss index 40aa5b53b..b27b5dfc3 100644 --- a/app/assets/stylesheets/tiny_mce.scss +++ b/app/assets/stylesheets/tiny_mce.scss @@ -66,4 +66,13 @@ .mce-toolbar { background: $color-white !important; } + +.restore-draft-notification { + background: $state-info-bg !important; + height: 25px !important; + padding-left: 10px !important; + padding-right: 10px !important; + padding-top: 5px !important; +} + // scss-lint:enable ImportantRule diff --git a/app/views/my_modules/_description_form.html.erb b/app/views/my_modules/_description_form.html.erb index 83abbc2bb..a8de52fcd 100644 --- a/app/views/my_modules/_description_form.html.erb +++ b/app/views/my_modules/_description_form.html.erb @@ -23,5 +23,7 @@ data: { object_type: 'my_module', object_id: @my_module.id, - highlightjs_path: asset_path('highlightjs-github-theme.css') } ) %> + highlightjs_path: asset_path('highlightjs-github-theme.css'), + last_updated: @my_module.updated_at.to_i * 1000 + } ) %> <% end %> From 4f2182e2f5b533b53c258ed04a62b19fe9aa9bfe Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Mon, 23 Sep 2019 22:57:47 +0200 Subject: [PATCH 02/11] Remove restoredraft icon --- app/assets/javascripts/sitewide/tiny_mce.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index 6cb71b88b..642aab446 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -92,7 +92,7 @@ var TinyMCE = (function() { cache_suffix: '?v=4.9.3', // This suffix should be changed any time library is updated selector: selector, menubar: 'file edit view insert format table', - toolbar: 'undo redo restoredraft | insert | styleselect | bold italic | alignleft aligncenter alignright alignjustify | bullist numlist outdent indent | link | forecolor backcolor | customimageuploader | codesample | table tabledelete | tableprops tablerowprops tablecellprops | tableinsertrowbefore tableinsertrowafter tabledeleterow | tableinsertcolbefore tableinsertcolafter tabledeletecol ', + toolbar: 'undo redo | insert | styleselect | bold italic | alignleft aligncenter alignright alignjustify | bullist numlist outdent indent | link | forecolor backcolor | customimageuploader | codesample | table tabledelete | tableprops tablerowprops tablecellprops | tableinsertrowbefore tableinsertrowafter tabledeleterow | tableinsertcolbefore tableinsertcolafter tabledeletecol', plugins: 'autosave autoresize customimageuploader link advlist codesample autolink lists charmap hr anchor searchreplace wordcount visualblocks visualchars insertdatetime nonbreaking save directionality paste textcolor placeholder colorpicker textpattern table', autoresize_bottom_margin: 20, codesample_languages: [ From b286e4adfda190d3fd038b0862bfd41138ec7386 Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Tue, 24 Sep 2019 09:24:16 +0200 Subject: [PATCH 03/11] Fix styles for notification bar --- app/assets/javascripts/sitewide/tiny_mce.js | 16 ++++++++++------ app/assets/stylesheets/tiny_mce.scss | 19 +++++++++++++++++++ 2 files changed, 29 insertions(+), 6 deletions(-) diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index 642aab446..2f36432d6 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -16,6 +16,12 @@ var TinyMCE = (function() { }); } + function refreshSaveButton(editor) { + var editorForm = $(editor.getContainer()).closest('form'); + editorForm.find('.tinymce-status-badge').addClass('hidden'); + $(editor.getContainer()) + .find('.tinymce-save-button').removeClass('hidden'); + } // Get LocalStorage auto save path @@ -36,8 +42,8 @@ var TinyMCE = (function() { var lastDraftTime = parseInt(tinyMCE.util.LocalStorage.getItem(prefix + 'time'), 10); var lastUpdated = $(selector).data('last-updated'); - var restoreBtn = $(''); - var cancelBtn = $('
' + + var restoreBtn = $(''); + var cancelBtn = $('
' + '
'); @@ -60,6 +66,7 @@ var TinyMCE = (function() { $(restoreBtn).click(function() { editor.plugins.autosave.restoreDraft(); editor.plugins.autosave.removeDraft(); + refreshSaveButton(editor); notificationBar.remove(); }); @@ -281,10 +288,7 @@ var TinyMCE = (function() { }); editor.on('Dirty', function() { - var editorForm = $(editor.getContainer()).closest('form'); - editorForm.find('.tinymce-status-badge').addClass('hidden'); - $(editor.getContainer()) - .find('.tinymce-save-button').removeClass('hidden'); + refreshSaveButton(editor); }); editor.on('remove', function() { diff --git a/app/assets/stylesheets/tiny_mce.scss b/app/assets/stylesheets/tiny_mce.scss index b27b5dfc3..3a4ddc868 100644 --- a/app/assets/stylesheets/tiny_mce.scss +++ b/app/assets/stylesheets/tiny_mce.scss @@ -67,12 +67,31 @@ background: $color-white !important; } +.tinymce-cancel-notification-button { + cursor: pointer; + + .fas { + color: $color-silver-chalice; + font-family: 'Font Awesome 5 Free'; + font-weight: 501; + margin-left: 10px; + margin-top: 4px; + } +} + .restore-draft-notification { background: $state-info-bg !important; height: 25px !important; padding-left: 10px !important; padding-right: 10px !important; padding-top: 5px !important; + + .restore-draft-btn { + border: 1px solid $color-silver-chalice; + font-size: 12px; + margin-top: -2px; + padding: 3px 10px 3px 10px; + } } // scss-lint:enable ImportantRule From 0a3637232141af4f810da1cd67d90ba2a1ac88d1 Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Tue, 24 Sep 2019 10:06:56 +0200 Subject: [PATCH 04/11] Fix placeholder positioning --- .../tinymce/plugins/placeholder/plugin.js | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/app/assets/javascripts/tinymce/plugins/placeholder/plugin.js b/app/assets/javascripts/tinymce/plugins/placeholder/plugin.js index be5c209bf..efae49143 100644 --- a/app/assets/javascripts/tinymce/plugins/placeholder/plugin.js +++ b/app/assets/javascripts/tinymce/plugins/placeholder/plugin.js @@ -27,12 +27,26 @@ tinymce.PluginManager.add('placeholder', function(editor) { editor.on('init', function() { var label = new Label(); + // Correct top css property due to notification bar + function calculatePlaceholderPosition() { + var editorForm = $(editor.getContainer()).closest('form'); + var editorToolbar = editorForm.find('.mce-top-part'); + + var restoreDraftNotification = $(editorForm).find('.restore-draft-notification'); + var restoreDraftHeight = restoreDraftNotification.context ? restoreDraftNotification.height() : 0; + var newTop = $(editorToolbar).height() + restoreDraftHeight; + $(label.el).css('top', newTop + 'px'); + } + function checkPlaceholder() { + // Show/hide depending on the content if (editor.getContent() === '') { label.show(); } else { label.hide(); } + + calculatePlaceholderPosition(); } function onKeydown() { From b3cafcec7c61b4194ea07978e964020f3b4d4463 Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Tue, 24 Sep 2019 11:26:36 +0200 Subject: [PATCH 05/11] Add restore draft notification to all models Close SCI-3751 --- app/assets/javascripts/sitewide/tiny_mce.js | 1 - app/views/my_modules/_description_form.html.erb | 3 +-- .../my_modules/protocols/_protocol_description_form.html.erb | 3 ++- app/views/result_texts/_edit.html.erb | 3 ++- app/views/result_texts/_new.html.erb | 3 ++- app/views/steps/_empty_step.html.erb | 3 ++- 6 files changed, 9 insertions(+), 7 deletions(-) diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index 2f36432d6..5d41d5b4e 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -65,7 +65,6 @@ var TinyMCE = (function() { $(restoreBtn).click(function() { editor.plugins.autosave.restoreDraft(); - editor.plugins.autosave.removeDraft(); refreshSaveButton(editor); notificationBar.remove(); }); diff --git a/app/views/my_modules/_description_form.html.erb b/app/views/my_modules/_description_form.html.erb index a8de52fcd..9750fa0cf 100644 --- a/app/views/my_modules/_description_form.html.erb +++ b/app/views/my_modules/_description_form.html.erb @@ -24,6 +24,5 @@ object_type: 'my_module', object_id: @my_module.id, highlightjs_path: asset_path('highlightjs-github-theme.css'), - last_updated: @my_module.updated_at.to_i * 1000 - } ) %> + last_updated: @my_module.updated_at.to_i * 1000 } ) %> <% end %> diff --git a/app/views/my_modules/protocols/_protocol_description_form.html.erb b/app/views/my_modules/protocols/_protocol_description_form.html.erb index fe8af693a..cc3f1180a 100644 --- a/app/views/my_modules/protocols/_protocol_description_form.html.erb +++ b/app/views/my_modules/protocols/_protocol_description_form.html.erb @@ -23,5 +23,6 @@ data: { object_type: 'protocol', object_id: protocol.id, - highlightjs_path: asset_path('highlightjs-github-theme.css') } ) %> + highlightjs_path: asset_path('highlightjs-github-theme.css'), + last_updated: protocol.updated_at.to_i * 1000 } ) %> <% end %> diff --git a/app/views/result_texts/_edit.html.erb b/app/views/result_texts/_edit.html.erb index e11c24eb6..636c66d16 100644 --- a/app/views/result_texts/_edit.html.erb +++ b/app/views/result_texts/_edit.html.erb @@ -7,7 +7,8 @@ value: @result.result_text.tinymce_render(:text), data: { object_type: 'result_text', object_id: @result.result_text.id, - highlightjs_path: asset_path('highlightjs-github-theme.css') }) %> + highlightjs_path: asset_path('highlightjs-github-theme.css'), + last_updated: @result.updated_at.to_i * 1000 }) %> <% end %>
From 289853e2e8f9f4a384d9c10712ed3821b15dfa9e Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Tue, 24 Sep 2019 14:04:07 +0200 Subject: [PATCH 06/11] Do not remove draft on step cancel (temporary) Close SCI-3751 --- app/assets/javascripts/protocols/steps.js.erb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/assets/javascripts/protocols/steps.js.erb b/app/assets/javascripts/protocols/steps.js.erb index 2cc5cf49b..916fac912 100644 --- a/app/assets/javascripts/protocols/steps.js.erb +++ b/app/assets/javascripts/protocols/steps.js.erb @@ -366,6 +366,7 @@ }); toggleButtons(true); DragNDropSteps.clearFiles(); + tinyMCE.editors.step_description_textarea.remove(); }); } @@ -604,7 +605,6 @@ toggleButtons(true); SmartAnnotation.preventPropagation('.atwho-user-popover'); - tinyMCE.editors.step_description_textarea.plugins.autosave.removeDraft(); tinyMCE.editors.step_description_textarea.remove(); //Rerender tables From 477e41a96d24fea1182c4607d4fb4e6807ec1e3b Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Tue, 24 Sep 2019 14:16:29 +0200 Subject: [PATCH 07/11] Refactor I18n strings for tiny_mce --- app/assets/javascripts/sitewide/tiny_mce.js | 4 ++-- config/locales/en.yml | 2 ++ 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index 5d41d5b4e..d0ba65e62 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -53,9 +53,9 @@ var TinyMCE = (function() { var notificationBar = $('
'); if (lastDraftTime < lastUpdated) { - notificationBar.text('Older version of the text below has been saved in the browser. Do you want to restore it?'); + notificationBar.text(I18n.t('tiny_mce.older_version_available')); } else { - notificationBar.text('Newer version of the text below has been saved in the browser. Do you want to restore it?'); + notificationBar.text(I18n.t('tiny_mce.newer_version_available')); } // Add notification bar diff --git a/config/locales/en.yml b/config/locales/en.yml index dd101cd70..09c9341d1 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -2007,6 +2007,8 @@ en: server_not_respond: "Didn't get a response from the server" saved_label: "Saved" leaving_warning: "You have made some changes, are you sure you want to leave this page?" + older_version_available: "Older version of the text below has been saved in the browser. Do you want to restore it?" + newer_version_available: "Newer version of the text below has been saved in the browser. Do you want to restore it?" general: save: "Save" update: "Update" From 8441c5b6ed62ea5fc691fbe1ac7b0a5141d737db Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Wed, 25 Sep 2019 18:08:25 +0200 Subject: [PATCH 08/11] Add anchor to result text editing --- app/assets/javascripts/sitewide/tiny_mce.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index d0ba65e62..76dcacec1 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -89,11 +89,11 @@ var TinyMCE = (function() { .before('
'); tinyMceContainer.addClass('hidden'); - if (textAreaObject.data('objectType') === 'step') { + if (textAreaObject.data('objectType') === 'step' + || textAreaObject.data('objectType') === 'result_text') { document.location.hash = textAreaObject.data('objectType') + '_' + textAreaObject.data('objectId'); } - tinyMCE.init({ cache_suffix: '?v=4.9.3', // This suffix should be changed any time library is updated selector: selector, From 070c307721313a619482ff4d683f43f7284378eb Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Mon, 30 Sep 2019 14:27:00 +0200 Subject: [PATCH 09/11] Implement style changes suggested by Anton --- app/assets/javascripts/protocols/steps.js.erb | 1 + app/assets/javascripts/sitewide/tiny_mce.js | 18 +++++++++--------- 2 files changed, 10 insertions(+), 9 deletions(-) diff --git a/app/assets/javascripts/protocols/steps.js.erb b/app/assets/javascripts/protocols/steps.js.erb index 916fac912..a78c48d95 100644 --- a/app/assets/javascripts/protocols/steps.js.erb +++ b/app/assets/javascripts/protocols/steps.js.erb @@ -605,6 +605,7 @@ toggleButtons(true); SmartAnnotation.preventPropagation('.atwho-user-popover'); + tinyMCE.editors.step_description_textarea.plugins.autosave.removeDraft(); tinyMCE.editors.step_description_textarea.remove(); //Rerender tables diff --git a/app/assets/javascripts/sitewide/tiny_mce.js b/app/assets/javascripts/sitewide/tiny_mce.js index 76dcacec1..e25701ebe 100644 --- a/app/assets/javascripts/sitewide/tiny_mce.js +++ b/app/assets/javascripts/sitewide/tiny_mce.js @@ -16,7 +16,7 @@ var TinyMCE = (function() { }); } - function refreshSaveButton(editor) { + function makeItDirty(editor) { var editorForm = $(editor.getContainer()).closest('form'); editorForm.find('.tinymce-status-badge').addClass('hidden'); $(editor.getContainer()) @@ -43,10 +43,11 @@ var TinyMCE = (function() { var lastUpdated = $(selector).data('last-updated'); var restoreBtn = $(''); - var cancelBtn = $('
' + - '
'); + var cancelBtn = $(`
+ +
`); // Check whether we have draft stored if (editor.plugins.autosave.hasDraft()) { @@ -59,13 +60,12 @@ var TinyMCE = (function() { } // Add notification bar - $(notificationBar).append(cancelBtn); - $(notificationBar).append(restoreBtn); + $(notificationBar).append(cancelBtn).append(restoreBtn); $(editor.contentAreaContainer).before(notificationBar); $(restoreBtn).click(function() { editor.plugins.autosave.restoreDraft(); - refreshSaveButton(editor); + makeItDirty(editor); notificationBar.remove(); }); @@ -287,7 +287,7 @@ var TinyMCE = (function() { }); editor.on('Dirty', function() { - refreshSaveButton(editor); + makeItDirty(editor); }); editor.on('remove', function() { From 096228abbbb145aa6ab812771f84256ac0e88c6e Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Mon, 30 Sep 2019 14:32:09 +0200 Subject: [PATCH 10/11] Shorten padding format in scss --- app/assets/stylesheets/tiny_mce.scss | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/app/assets/stylesheets/tiny_mce.scss b/app/assets/stylesheets/tiny_mce.scss index 3a4ddc868..c85b87a27 100644 --- a/app/assets/stylesheets/tiny_mce.scss +++ b/app/assets/stylesheets/tiny_mce.scss @@ -82,9 +82,7 @@ .restore-draft-notification { background: $state-info-bg !important; height: 25px !important; - padding-left: 10px !important; - padding-right: 10px !important; - padding-top: 5px !important; + padding: 5px 10px 0px 10px !important; .restore-draft-btn { border: 1px solid $color-silver-chalice; From 9f704559efb4dc59e63c6b78ff603fbc75f06f44 Mon Sep 17 00:00:00 2001 From: Jure Grabnar Date: Mon, 30 Sep 2019 14:42:15 +0200 Subject: [PATCH 11/11] Fix hound formatting issues --- app/assets/stylesheets/tiny_mce.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/assets/stylesheets/tiny_mce.scss b/app/assets/stylesheets/tiny_mce.scss index c85b87a27..cc8101097 100644 --- a/app/assets/stylesheets/tiny_mce.scss +++ b/app/assets/stylesheets/tiny_mce.scss @@ -82,7 +82,7 @@ .restore-draft-notification { background: $state-info-bg !important; height: 25px !important; - padding: 5px 10px 0px 10px !important; + padding: 5px 10px 0 !important; .restore-draft-btn { border: 1px solid $color-silver-chalice;