mirror of
https://github.com/documize/community.git
synced 2025-07-24 15:49:44 +02:00
Introduce new Tabular editor with CSV import support
Closes #211 and #202 An all-new tabular editor has been added -- this replaces the previous tabular editor. Better formatting options. CSV data can also be imported straight into the table.
This commit is contained in:
parent
ed99b0c9f3
commit
25c247e99b
18 changed files with 2224 additions and 14 deletions
|
@ -24,6 +24,8 @@ export default Component.extend({
|
|||
defaultTable: '<table class="wysiwyg-table" style="width: 100%;"><thead><tr><th><br></th><th><br></th><th><br></th><th><br></th></tr></thead><tbody><tr><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td></tr><tr><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td></tr><tr><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td></tr></tbody></table>',
|
||||
|
||||
didReceiveAttrs() {
|
||||
this._super(...arguments);
|
||||
|
||||
this.set('pageBody', this.get('meta.rawBody'));
|
||||
|
||||
if (is.empty(this.get('pageBody'))) {
|
||||
|
@ -32,7 +34,10 @@ export default Component.extend({
|
|||
},
|
||||
|
||||
didInsertElement() {
|
||||
this._super(...arguments);
|
||||
|
||||
let id = '#' + this.get('editorId');
|
||||
|
||||
$(id).froalaEditor({
|
||||
toolbarButtons: [],
|
||||
toolbarInline: true,
|
||||
|
@ -47,6 +52,8 @@ export default Component.extend({
|
|||
},
|
||||
|
||||
willDestroyElement() {
|
||||
this._super(...arguments);
|
||||
|
||||
$('#' + this.get('editorId')).off('froalaEditor.contentChanged');
|
||||
},
|
||||
|
||||
|
|
245
gui/app/components/section/tabular/type-editor.js
Normal file
245
gui/app/components/section/tabular/type-editor.js
Normal file
|
@ -0,0 +1,245 @@
|
|||
// Copyright 2016 Documize Inc. <legal@documize.com>. All rights reserved.
|
||||
//
|
||||
// This software (Documize Community Edition) is licensed under
|
||||
// GNU AGPL v3 http://www.gnu.org/licenses/agpl-3.0.en.html
|
||||
//
|
||||
// You can operate outside the AGPL restrictions by purchasing
|
||||
// Documize Enterprise Edition and obtaining a commercial license
|
||||
// by contacting <sales@documize.com>.
|
||||
//
|
||||
// https://documize.com
|
||||
|
||||
import $ from 'jquery';
|
||||
import { computed, set } from '@ember/object';
|
||||
import { schedule } from '@ember/runloop';
|
||||
import { inject as service } from '@ember/service';
|
||||
import Modals from '../../../mixins/modal';
|
||||
import Notifier from '../../../mixins/notifier';
|
||||
import Component from '@ember/component';
|
||||
|
||||
export default Component.extend(Modals, Notifier, {
|
||||
appMeta: service(),
|
||||
link: service(),
|
||||
pageBody: '',
|
||||
editorId: computed('page', function () {
|
||||
let page = this.get('page');
|
||||
return `tabular-editor-${page.id}`;
|
||||
}),
|
||||
modalId: computed('page', function () {
|
||||
let page = this.get('page');
|
||||
return `tabular-editor-modal-${page.id}`;
|
||||
}),
|
||||
importData: '',
|
||||
importOption: null,
|
||||
defaultTable: '<table class="wysiwyg-table" style="width: 100%;"><thead><tr><th><br></th><th><br></th><th><br></th><th><br></th></tr></thead><tbody><tr><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td></tr><tr><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td></tr><tr><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td><td style="width: 25.0000%;"><br></td></tr></tbody></table>',
|
||||
|
||||
didReceiveAttrs() {
|
||||
this._super(...arguments);
|
||||
|
||||
this.set('pageBody', this.get('meta.rawBody'));
|
||||
|
||||
if (is.empty(this.get('pageBody'))) {
|
||||
this.set('pageBody', this.get('defaultTable'));
|
||||
}
|
||||
|
||||
this.set('importOption', {
|
||||
headers: true,
|
||||
parseTypes: false,
|
||||
});
|
||||
},
|
||||
|
||||
|
||||
didInsertElement() {
|
||||
this._super(...arguments);
|
||||
|
||||
this.addEditor();
|
||||
},
|
||||
|
||||
willDestroyElement() {
|
||||
this._super(...arguments);
|
||||
|
||||
this.removeEditor();
|
||||
},
|
||||
|
||||
addEditor() {
|
||||
schedule('afterRender', () => {
|
||||
let options = {
|
||||
cache_suffix: '?v=492',
|
||||
selector: '#' + this.get('editorId'),
|
||||
relative_urls: false,
|
||||
browser_spellcheck: true,
|
||||
gecko_spellcheck: false,
|
||||
statusbar: false,
|
||||
inline: true,
|
||||
paste_data_images: true,
|
||||
images_upload_handler: function (blobInfo, success, failure) { // eslint-disable-line no-unused-vars
|
||||
success("data:" + blobInfo.blob().type + ";base64," + blobInfo.base64());
|
||||
},
|
||||
image_advtab: true,
|
||||
media_live_embeds: true,
|
||||
theme: 'modern',
|
||||
skin: 'lightgray-gradient',
|
||||
entity_encoding: 'raw',
|
||||
extended_valid_elements: 'b,i,b/strong,i/em',
|
||||
fontsize_formats:
|
||||
'8px 10px 12px 14px 15px 16px 18px 20px 22px 24px 26px 28px 30px 32px 34px 36px 38px 40px 42px 44px 46px 48px 50px 52px 54px 56px 58px 60px 70px 80px 90px 100px',
|
||||
formats: {
|
||||
bold: {
|
||||
inline: 'b'
|
||||
},
|
||||
italic: {
|
||||
inline: 'i'
|
||||
}
|
||||
},
|
||||
plugins: [
|
||||
'advlist autolink lists link image charmap print hr anchor pagebreak',
|
||||
'searchreplace wordcount visualblocks visualchars code codesample fullscreen',
|
||||
'insertdatetime media nonbreaking save table directionality',
|
||||
'template paste textcolor colorpicker textpattern imagetools uploadimage'
|
||||
],
|
||||
menu: {},
|
||||
menubar: false,
|
||||
table_toolbar: '',
|
||||
toolbar1: 'table tabledelete | tableprops tablerowprops tablecellprops | tableinsertrowbefore tableinsertrowafter tabledeleterow | tableinsertcolbefore tableinsertcolafter tabledeletecol',
|
||||
toolbar2: 'fontsizeselect | forecolor backcolor link unlink | bold italic underline strikethrough | alignleft aligncenter alignright alignjustify',
|
||||
save_onsavecallback: function () {
|
||||
Mousetrap.trigger('ctrl+s');
|
||||
}
|
||||
};
|
||||
|
||||
if (typeof tinymce === 'undefined') {
|
||||
$.getScript('/tinymce/tinymce.min.js?v=492', function () {
|
||||
window.tinymce.dom.Event.domLoaded = true;
|
||||
tinymce.baseURL = '//' + window.location.host + '/tinymce';
|
||||
tinymce.suffix = '.min';
|
||||
tinymce.init(options);
|
||||
});
|
||||
} else {
|
||||
tinymce.init(options);
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
removeEditor() {
|
||||
tinymce.EditorManager.execCommand('mceRemoveEditor', true, this.get('editorId'));
|
||||
},
|
||||
|
||||
generateImportTable(results) {
|
||||
// nothing to import?
|
||||
if (is.undefined(results) || results.data.length === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
let opts = this.get('importOption');
|
||||
|
||||
let table = '<table class="wysiwyg-table" style="width: 100%;"><thead><tr>';
|
||||
|
||||
// Setup the table headers
|
||||
if (opts.headers && is.array(results.meta.fields)) {
|
||||
// use headers from file
|
||||
results.meta.fields.forEach((header) => {
|
||||
table = table + '<th>' + header.trim() + '</th>';
|
||||
});
|
||||
} else {
|
||||
// create dummy headers
|
||||
for (let i = 1; i <= results.data[0].length; i++) {
|
||||
table = table + '<th>Column ' + i + '</th>';
|
||||
}
|
||||
}
|
||||
|
||||
table = table + '</tr></thead>'
|
||||
|
||||
// now convert data rows to table.
|
||||
table = table + '<tbody>'
|
||||
|
||||
results.data.forEach(row => {
|
||||
table = table + '<tr>';
|
||||
|
||||
if (is.array(row)) {
|
||||
row.forEach((cell) => {
|
||||
table = table + '<td>' + cell.trim() + '</td>';
|
||||
});
|
||||
} else {
|
||||
// convert Javascript object to array
|
||||
let cells = Object.values(row);
|
||||
|
||||
cells.forEach((cell) => {
|
||||
table = table + '<td>' + cell.trim() + '</td>';
|
||||
});
|
||||
}
|
||||
|
||||
table = table + '</tr>'
|
||||
});
|
||||
|
||||
table = table + '</tbody>'
|
||||
table = table + '</table>';
|
||||
|
||||
let editor = tinymce.EditorManager.get(this.get('editorId'));
|
||||
editor.setContent(table);
|
||||
},
|
||||
|
||||
actions: {
|
||||
onShowImportModal() {
|
||||
this.modalOpen('#' + this.get('modalId'), {show:true}, "#csv-data");
|
||||
},
|
||||
|
||||
onImport() {
|
||||
let csv = this.get('importData');
|
||||
let opts = this.get('importOption');
|
||||
|
||||
this.modalClose('#' + this.get('modalId'));
|
||||
|
||||
if (is.empty(csv)) {
|
||||
return;
|
||||
}
|
||||
|
||||
let results = Papa.parse(csv, {
|
||||
header: opts.headers,
|
||||
dynamicTyping: opts.parseTypes,
|
||||
skipEmptyLines: true,
|
||||
});
|
||||
|
||||
this.generateImportTable(results);
|
||||
},
|
||||
|
||||
onInsertLink(link) {
|
||||
let editor = tinymce.EditorManager.get(this.get('editorId'));
|
||||
let userSelection = editor.selection.getContent();
|
||||
|
||||
if (is.not.empty(userSelection)) {
|
||||
set(link, 'title', userSelection);
|
||||
}
|
||||
|
||||
let linkHTML = this.get('link').buildLink(link);
|
||||
editor.insertContent(linkHTML);
|
||||
|
||||
return true;
|
||||
},
|
||||
|
||||
isDirty() {
|
||||
let editor = tinymce.EditorManager.get(this.get('editorId'));
|
||||
return (
|
||||
is.not.undefined(tinymce) &&
|
||||
is.not.undefined(editor) &&
|
||||
editor.isDirty()
|
||||
);
|
||||
},
|
||||
|
||||
onCancel() {
|
||||
let cb = this.get('onCancel');
|
||||
cb();
|
||||
},
|
||||
|
||||
onAction(title) {
|
||||
let page = this.get('page');
|
||||
let meta = this.get('meta');
|
||||
let editor = tinymce.EditorManager.get(this.get('editorId'));
|
||||
|
||||
page.set('title', title);
|
||||
meta.set('rawBody', editor.getContent());
|
||||
|
||||
let cb = this.get('onAction');
|
||||
cb(page, meta);
|
||||
}
|
||||
}
|
||||
});
|
14
gui/app/components/section/tabular/type-renderer.js
Normal file
14
gui/app/components/section/tabular/type-renderer.js
Normal file
|
@ -0,0 +1,14 @@
|
|||
// Copyright 2016 Documize Inc. <legal@documize.com>. All rights reserved.
|
||||
//
|
||||
// This software (Documize Community Edition) is licensed under
|
||||
// GNU AGPL v3 http://www.gnu.org/licenses/agpl-3.0.en.html
|
||||
//
|
||||
// You can operate outside the AGPL restrictions by purchasing
|
||||
// Documize Enterprise Edition and obtaining a commercial license
|
||||
// by contacting <sales@documize.com>.
|
||||
//
|
||||
// https://documize.com
|
||||
|
||||
import Component from '@ember/component';
|
||||
|
||||
export default Component.extend({ });
|
|
@ -107,7 +107,7 @@ export default Component.extend({
|
|||
],
|
||||
menu: {},
|
||||
menubar: false,
|
||||
toolbar1: 'formatselect fontsizeselect | bold italic underline strikethrough superscript subscript | forecolor backcolor link unlink',
|
||||
toolbar1: 'formatselect fontsizeselect | bold italic underline strikethrough superscript subscript blockquote | forecolor backcolor link unlink',
|
||||
toolbar2: 'outdent indent bullist numlist | alignleft aligncenter alignright alignjustify | table uploadimage image media codesample',
|
||||
save_onsavecallback: function () {
|
||||
Mousetrap.trigger('ctrl+s');
|
||||
|
@ -129,11 +129,8 @@ export default Component.extend({
|
|||
|
||||
willDestroyElement() {
|
||||
this._super(...arguments);
|
||||
tinymce.EditorManager.execCommand(
|
||||
'mceRemoveEditor',
|
||||
true,
|
||||
this.get('editorId')
|
||||
);
|
||||
|
||||
tinymce.EditorManager.execCommand('mceRemoveEditor', true, this.get('editorId'));
|
||||
},
|
||||
|
||||
actions: {
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue