From d937f31f394e5b10a84be6fb6ebbd02dceabc0b3 Mon Sep 17 00:00:00 2001 From: "phuc.nt" Date: Tue, 27 Feb 2018 09:31:21 +0700 Subject: [PATCH] [10.0][MIG] web_widget_text_markdown Adjustment to make it work version 10.0 --- web_widget_text_markdown/README.rst | 15 +- web_widget_text_markdown/__manifest__.py | 31 +- .../demo/bootstrap_markdown.xml | 12 + .../bootstrap-markdown/bootstrap-markdown.js | 1641 ++++++++++------- .../static/src/css/main.css | 4 +- .../static/src/js/web_widget_text_markdown.js | 51 +- .../static/src/xml/bootstrap_markdown.xml | 10 +- web_widget_text_markdown/views/main.xml | 13 +- 8 files changed, 1018 insertions(+), 759 deletions(-) create mode 100644 web_widget_text_markdown/demo/bootstrap_markdown.xml diff --git a/web_widget_text_markdown/README.rst b/web_widget_text_markdown/README.rst index 4ce76627..72c4306a 100644 --- a/web_widget_text_markdown/README.rst +++ b/web_widget_text_markdown/README.rst @@ -8,11 +8,6 @@ This module adds a new widget for text field in form view on Odoo: [1]: http://www.codingdrama.com/bootstrap-markdown/ "bootstrap-markdown" -Installation -============ - -It was tested on openerp trunk, 8.0 branch. - Usage ===== @@ -48,6 +43,16 @@ Contributors ------------ * Nicolas Jeudy +* Nguyen Tan Phuc + +Do not contact contributors directly about support or help with technical issues. + +Funders +------- + +The development of this module has been financially supported by: + +* Komit https://komit-consulting.com Maintainer ---------- diff --git a/web_widget_text_markdown/__manifest__.py b/web_widget_text_markdown/__manifest__.py index f3678db2..e281a0e6 100644 --- a/web_widget_text_markdown/__manifest__.py +++ b/web_widget_text_markdown/__manifest__.py @@ -2,6 +2,7 @@ ############################################################################## # # Copyright (C) 2014 Sudokeys () +# Copyright (C) 2017 Komit () # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published @@ -20,18 +21,26 @@ { 'name': 'web_widget_text_markdown', - 'version': '8.0.1.0.0', - 'author': "Sudokeys,Odoo Community Association (OCA)", - 'maintainer': 'Sudokeys', - 'category': '', + 'version': '10.0.1.0.0', + "author": "Komit, " + "Sudokeys, " + "Odoo Community Association (OCA)", + 'category': 'Web', 'license': 'AGPL-3', - 'depends': ['base', 'web'], - 'website': 'http://www.sudokey.com', - 'data': ['views/main.xml', ], - "qweb": ["static/src/xml/bootstrap_markdown.xml", - ], - 'demo': [], - 'installable': False, + 'website': 'https://github.com/OCA/web', + 'depends': [ + 'base', 'web' + ], + 'demo': [ + "demo/bootstrap_markdown.xml", + ], + 'data': [ + 'views/main.xml', + ], + "qweb": [ + "static/src/xml/bootstrap_markdown.xml", + ], + 'installable': True, 'auto_install': False, 'application': False } diff --git a/web_widget_text_markdown/demo/bootstrap_markdown.xml b/web_widget_text_markdown/demo/bootstrap_markdown.xml new file mode 100644 index 00000000..0442662e --- /dev/null +++ b/web_widget_text_markdown/demo/bootstrap_markdown.xml @@ -0,0 +1,12 @@ + + + + res.groups + + + + bootstrap_markdown + + + + diff --git a/web_widget_text_markdown/static/lib/bootstrap-markdown/bootstrap-markdown.js b/web_widget_text_markdown/static/lib/bootstrap-markdown/bootstrap-markdown.js index 66ad62f1..fffdbcf1 100644 --- a/web_widget_text_markdown/static/lib/bootstrap-markdown/bootstrap-markdown.js +++ b/web_widget_text_markdown/static/lib/bootstrap-markdown/bootstrap-markdown.js @@ -1,8 +1,8 @@ /* =================================================== - * bootstrap-markdown.js v2.7.0 + * bootstrap-markdown.js v2.10.0 * http://github.com/toopay/bootstrap-markdown * =================================================== - * Copyright 2013-2014 Taufan Aditya + * Copyright 2013-2016 Taufan Aditya * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,97 +16,125 @@ * See the License for the specific language governing permissions and * limitations under the License. * ========================================================== */ - -!function ($) { - - "use strict"; // jshint ;_; - +(function(factory) { + if (typeof define === "function" && define.amd) { + // RequireJS + define(["jquery"], factory); + } else if (typeof exports === 'object') { + // Backbone.js + factory(require('jquery')); + } else { + // jQuery plugin + factory(jQuery); + } +}(function($) { + "use strict"; /* MARKDOWN CLASS DEFINITION * ========================== */ - var Markdown = function (element, options) { + var Markdown = function(element, options) { + // @TODO : remove this BC on next major release + // @see : https://github.com/toopay/bootstrap-markdown/issues/109 + var opts = ['autofocus', 'savable', 'hideable', 'width', + 'height', 'resize', 'iconlibrary', 'language', + 'footer', 'fullscreen', 'hiddenButtons', 'disabledButtons' + ]; + $.each(opts, function(_, opt) { + if (typeof $(element).data(opt) !== 'undefined') { + options = typeof options == 'object' ? options : {}; + options[opt] = $(element).data(opt); + } + }); + // End BC + // Class Properties - this.$ns = 'bootstrap-markdown' - this.$element = $(element) - this.$editable = {el:null, type:null,attrKeys:[], attrValues:[], content:null} - this.$options = $.extend(true, {}, $.fn.markdown.defaults, options, this.$element.data(), this.$element.data('options')) - this.$oldContent = null - this.$isPreview = false - this.$isFullscreen = false - this.$editor = null - this.$textarea = null - this.$handler = [] - this.$callback = [] - this.$nextTab = [] - - this.showEditor() - } + this.$ns = 'bootstrap-markdown'; + this.$element = $(element); + this.$editable = { + el: null, + type: null, + attrKeys: [], + attrValues: [], + content: null + }; + this.$options = $.extend(true, {}, $.fn.markdown.defaults, options, this.$element.data('options')); + this.$oldContent = null; + this.$isPreview = false; + this.$isFullscreen = false; + this.$editor = null; + this.$textarea = null; + this.$handler = []; + this.$callback = []; + this.$nextTab = []; + + this.showEditor(); + }; Markdown.prototype = { - constructor: Markdown + constructor: Markdown, + __alterButtons: function(name, alter) { + var handler = this.$handler, + isAll = (name == 'all'), + that = this; - , __alterButtons: function(name,alter) { - var handler = this.$handler, isAll = (name == 'all'),that = this - - $.each(handler,function(k,v) { - var halt = true + $.each(handler, function(k, v) { + var halt = true; if (isAll) { - halt = false + halt = false; } else { - halt = v.indexOf(name) < 0 + halt = v.indexOf(name) < 0; } - if (halt == false) { - alter(that.$editor.find('button[data-handler="'+v+'"]')) + if (halt === false) { + alter(that.$editor.find('button[data-handler="' + v + '"]')); } - }) - } - - , __buildButtons: function(buttonsArray, container) { + }); + }, + __buildButtons: function(buttonsArray, container) { var i, - ns = this.$ns, - handler = this.$handler, - callback = this.$callback + ns = this.$ns, + handler = this.$handler, + callback = this.$callback; - for (i=0;i', { - 'class': 'btn-group' - }) + buttons = btnGroups[y].data, + btnGroupContainer = $('
', { + 'class': 'btn-group' + }); - for (z=0;z'); buttonContainer.text(' ' + this.__localize(btnText)).addClass('btn-default btn-sm').addClass(btnClass); - if(btnClass.match(/btn\-(primary|success|info|warning|danger|link)/)){ - buttonContainer.removeClass('btn-default'); + if (btnClass.match(/btn\-(primary|success|info|warning|danger|link)/)) { + buttonContainer.removeClass('btn-default'); } buttonContainer.attr({ - 'type': 'button', - 'title': this.__localize(button.title) + hotkeyCaption, - 'tabindex': tabIndex, - 'data-provider': ns, - 'data-handler': buttonHandler, - 'data-hotkey': hotkey + 'type': 'button', + 'title': this.__localize(button.title) + hotkeyCaption, + 'tabindex': tabIndex, + 'data-provider': ns, + 'data-handler': buttonHandler, + 'data-hotkey': hotkey }); - if (button.toggle == true){ + if (button.toggle === true) { buttonContainer.attr('data-toggle', 'button'); } buttonIconContainer = $(''); @@ -121,50 +149,55 @@ callback.push(button.callback); } - // Attach the button group into container dom + // Attach the button group into container DOM container.append(btnGroupContainer); } } return container; - } - , __setListener: function() { + }, + __setListener: function() { // Set size and resizable Properties - var hasRows = typeof this.$textarea.attr('rows') != 'undefined', - maxRows = this.$textarea.val().split("\n").length > 5 ? this.$textarea.val().split("\n").length : '5', - rowsVal = hasRows ? this.$textarea.attr('rows') : maxRows + var hasRows = typeof this.$textarea.attr('rows') !== 'undefined', + maxRows = this.$textarea.val().split("\n").length > 5 ? this.$textarea.val().split("\n").length : '5', + rowsVal = hasRows ? this.$textarea.attr('rows') : maxRows; - this.$textarea.attr('rows',rowsVal) + this.$textarea.attr('rows', rowsVal); if (this.$options.resize) { - this.$textarea.css('resize',this.$options.resize) + this.$textarea.css('resize', this.$options.resize); } - this.$textarea - .on('focus', $.proxy(this.focus, this)) - .on('keypress', $.proxy(this.keypress, this)) - .on('keyup', $.proxy(this.keyup, this)) - .on('change', $.proxy(this.change, this)) + // Re-attach markdown data + this.$textarea.data('markdown', this); + }, + __setEventListeners: function() { + this.$textarea.on({ + 'focus': $.proxy(this.focus, this), + 'keyup': $.proxy(this.keyup, this), + 'change': $.proxy(this.change, this), + 'select': $.proxy(this.select, this) + }); if (this.eventSupported('keydown')) { - this.$textarea.on('keydown', $.proxy(this.keydown, this)) + this.$textarea.on('keydown', $.proxy(this.keydown, this)); } - // Re-attach markdown data - this.$textarea.data('markdown',this) - } - - , __handle: function(e) { + if (this.eventSupported('keypress')) { + this.$textarea.on('keypress', $.proxy(this.keypress, this)); + } + }, + __handle: function(e) { var target = $(e.currentTarget), - handler = this.$handler, - callback = this.$callback, - handlerName = target.attr('data-handler'), - callbackIndex = handler.indexOf(handlerName), - callbackHandler = callback[callbackIndex] + handler = this.$handler, + callback = this.$callback, + handlerName = target.attr('data-handler'), + callbackIndex = handler.indexOf(handlerName), + callbackHandler = callback[callbackIndex]; // Trigger the focusin - $(e.currentTarget).focus() + $(e.currentTarget).focus(); - callbackHandler(this) + callbackHandler(this); // Trigger onChange for each button handle this.change(this); @@ -172,15 +205,14 @@ // Unless it was the save handler, // focusin the textarea if (handlerName.indexOf('cmdSave') < 0) { - this.$textarea.focus() + this.$textarea.focus(); } - e.preventDefault() - } - - , __localize: function(string) { + e.preventDefault(); + }, + __localize: function(string) { var messages = $.fn.markdown.messages, - language = this.$options.language + language = this.$options.language; if ( typeof messages !== 'undefined' && typeof messages[language] !== 'undefined' && @@ -189,185 +221,209 @@ return messages[language][string]; } return string; - } - - , __getIcon: function(src) { - return typeof src == 'object' ? src[this.$options.iconlibrary] : src; - } - - , setFullscreen: function(mode) { - var $editor = this.$editor, - $textarea = this.$textarea - - if (mode === true) { - $editor.addClass('md-fullscreen-mode') - $('body').addClass('md-nooverflow') - this.$options.onFullscreen(this) - } else { - $editor.removeClass('md-fullscreen-mode') - $('body').removeClass('md-nooverflow') - } + }, + __getIcon: function(src) { + if(typeof src == 'object'){ + var customIcon = this.$options.customIcons[src.name]; + return typeof customIcon == 'undefined' ? src.icon[this.$options.iconlibrary] : customIcon; + } else { + return src; + } + }, + setFullscreen: function(mode) { + var $editor = this.$editor, + $textarea = this.$textarea; + + if (mode === true) { + $editor.addClass('md-fullscreen-mode'); + $('body').addClass('md-nooverflow'); + this.$options.onFullscreen(this); + } else { + $editor.removeClass('md-fullscreen-mode'); + $('body').removeClass('md-nooverflow'); + this.$options.onFullscreenExit(this); - this.$isFullscreen = mode; - $textarea.focus() - } + if (this.$isPreview === true) + this.hidePreview().showPreview(); + } - , showEditor: function() { + this.$isFullscreen = mode; + $textarea.focus(); + }, + showEditor: function() { var instance = this, - textarea, - ns = this.$ns, - container = this.$element, - originalHeigth = container.css('height'), - originalWidth = container.css('width'), - editable = this.$editable, - handler = this.$handler, - callback = this.$callback, - options = this.$options, - editor = $( '
', { - 'class': 'md-editor', - click: function() { - instance.focus() - } - }) + textarea, + ns = this.$ns, + container = this.$element, + originalHeigth = container.css('height'), + originalWidth = container.css('width'), + editable = this.$editable, + handler = this.$handler, + callback = this.$callback, + options = this.$options, + editor = $('
', { + 'class': 'md-editor', + click: function() { + instance.focus(); + } + }); // Prepare the editor - if (this.$editor == null) { + if (this.$editor === null) { // Create the panel var editorHeader = $('
', { - 'class': 'md-header btn-toolbar' - }) + 'class': 'md-header btn-toolbar' + }); // Merge the main & additional button groups together - var allBtnGroups = [] - if (options.buttons.length > 0) allBtnGroups = allBtnGroups.concat(options.buttons[0]) - if (options.additionalButtons.length > 0) allBtnGroups = allBtnGroups.concat(options.additionalButtons[0]) + var allBtnGroups = []; + if (options.buttons.length > 0) allBtnGroups = allBtnGroups.concat(options.buttons[0]); + if (options.additionalButtons.length > 0) { + // iterate the additional button groups + $.each(options.additionalButtons[0], function(idx, buttonGroup) { + + // see if the group name of the additional group matches an existing group + var matchingGroups = $.grep(allBtnGroups, function(allButtonGroup, allIdx) { + return allButtonGroup.name === buttonGroup.name; + }); + + // if it matches add the additional buttons to that group, if not just add it to the all buttons group + if (matchingGroups.length > 0) { + matchingGroups[0].data = matchingGroups[0].data.concat(buttonGroup.data); + } else { + allBtnGroups.push(options.additionalButtons[0][idx]); + } + + }); + } // Reduce and/or reorder the button groups if (options.reorderButtonGroups.length > 0) { allBtnGroups = allBtnGroups - .filter(function(btnGroup) { - return options.reorderButtonGroups.indexOf(btnGroup.name) > -1 - }) - .sort(function(a, b) { - if (options.reorderButtonGroups.indexOf(a.name) < options.reorderButtonGroups.indexOf(b.name)) return -1 - if (options.reorderButtonGroups.indexOf(a.name) > options.reorderButtonGroups.indexOf(b.name)) return 1 - return 0 - }) + .filter(function(btnGroup) { + return options.reorderButtonGroups.indexOf(btnGroup.name) > -1; + }) + .sort(function(a, b) { + if (options.reorderButtonGroups.indexOf(a.name) < options.reorderButtonGroups.indexOf(b.name)) return -1; + if (options.reorderButtonGroups.indexOf(a.name) > options.reorderButtonGroups.indexOf(b.name)) return 1; + return 0; + }); } // Build the buttons if (allBtnGroups.length > 0) { - editorHeader = this.__buildButtons([allBtnGroups], editorHeader) + editorHeader = this.__buildButtons([allBtnGroups], editorHeader); } if (options.fullscreen.enable) { - editorHeader.append('
').on('click', '.md-control-fullscreen', function(e) { - e.preventDefault(); - instance.setFullscreen(true) - }) + editorHeader.append('
').on('click', '.md-control-fullscreen', function(e) { + e.preventDefault(); + instance.setFullscreen(true); + }); } - editor.append(editorHeader) + editor.append(editorHeader); // Wrap the textarea if (container.is('textarea')) { - container.before(editor) - textarea = container - textarea.addClass('md-input') - editor.append(textarea) + container.before(editor); + textarea = container; + textarea.addClass('md-input'); + editor.append(textarea); } else { var rawContent = (typeof toMarkdown == 'function') ? toMarkdown(container.html()) : container.html(), - currentContent = $.trim(rawContent) + currentContent = $.trim(rawContent); // This is some arbitrary content that could be edited textarea = $(' - +
diff --git a/web_widget_text_markdown/views/main.xml b/web_widget_text_markdown/views/main.xml index e8a8ab26..73ab4309 100644 --- a/web_widget_text_markdown/views/main.xml +++ b/web_widget_text_markdown/views/main.xml @@ -1,6 +1,7 @@ - +