From a670f092b936f41c3d1ced23bf62031ce185ea2b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9bastien=20Theys?= <seb@odoo.com> Date: Thu, 13 Sep 2018 18:15:01 +0200 Subject: [PATCH] [IMP] website: improve Optimize SEO and Page Manager Modal generic: - Fix footer overflowing buttons by pushing them to a new line. Page manager: - Add an SEO column to easily see which pages have incomplete SEO. - Add an edit SEO column to navigate to the SEO modal on said page. - Hide "edit in backend" column as debug. - Generally improve the whole view by adding appropriate titles on icons, etc. SEO modal: - Improve general style. - Add description length alert. - Restructure the keyword table/suggestions. - Reset page meta to their initial value on Discard. - Various small usability improvements. PR: #26897 task-1850579 --- addons/web/static/src/scss/modal.scss | 5 + addons/website/models/website.py | 6 + addons/website/static/src/js/menu/seo.js | 541 +++++++++++------- .../website/static/src/scss/website.ui.scss | 120 +--- addons/website/static/src/xml/website.seo.xml | 179 +++--- addons/website/views/website_templates.xml | 69 ++- 6 files changed, 487 insertions(+), 433 deletions(-) diff --git a/addons/web/static/src/scss/modal.scss b/addons/web/static/src/scss/modal.scss index c525050e1ec6..f02765c25c4f 100644 --- a/addons/web/static/src/scss/modal.scss +++ b/addons/web/static/src/scss/modal.scss @@ -26,6 +26,7 @@ } .modal-footer { + flex-wrap: wrap; text-align: left; justify-content: flex-start; @@ -39,6 +40,10 @@ > :not(:first-child) { margin-left: .25rem; } > :not(:last-child) { margin-right: .25rem; } } + + button { + margin-bottom: .5rem; + } } } diff --git a/addons/website/models/website.py b/addons/website/models/website.py index 85c7a9184234..7504fa5c2ec4 100644 --- a/addons/website/models/website.py +++ b/addons/website/models/website.py @@ -676,11 +676,17 @@ class SeoMetadata(models.AbstractModel): _name = 'website.seo.metadata' _description = 'SEO metadata' + is_seo_optimized = fields.Boolean("SEO optimized", compute='_compute_is_seo_optimized') website_meta_title = fields.Char("Website meta title", translate=True) website_meta_description = fields.Text("Website meta description", translate=True) website_meta_keywords = fields.Char("Website meta keywords", translate=True) website_meta_og_img = fields.Char("Website opengraph image") + @api.multi + def _compute_is_seo_optimized(self): + for record in self: + record.is_seo_optimized = record.website_meta_title and record.website_meta_description and record.website_meta_keywords + def _default_website_meta(self): """ This method will return default meta information. It return the dict contains meta property as a key and meta content as a value. diff --git a/addons/website/static/src/js/menu/seo.js b/addons/website/static/src/js/menu/seo.js index 0f1c268dbb86..4267cad2bbfe 100644 --- a/addons/website/static/src/js/menu/seo.js +++ b/addons/website/static/src/js/menu/seo.js @@ -17,22 +17,6 @@ var _t = core._t; // Javascript \b is not unicode aware, and words beginning or ending by accents won't match \b var WORD_SEPARATORS_REGEX = '([\\u2000-\\u206F\\u2E00-\\u2E7F\'!"#\\$%&\\(\\)\\*\\+,\\-\\.\\/:;<=>\\?¿¡@\\[\\]\\^_`\\{\\|\\}~\\s]+|^|$)'; -function analyzeKeyword(htmlPage, keyword) { - return htmlPage.isInTitle(keyword) ? { - title: 'badge badge-success', - description: "This keyword is used in the page title", - } : htmlPage.isInDescription(keyword) ? { - title: 'badge badge-primary', - description: "This keyword is used in the page description", - } : htmlPage.isInBody(keyword) ? { - title: 'badge badge-info', - description: "This keyword is used in the page content." - } : { - title: 'badge badge-secondary', - description: "This keyword is not used anywhere on the page." - }; -} - var Suggestion = Widget.extend({ template: 'website.seo_suggestion', xmlDependencies: ['/website/static/src/xml/website.seo.xml'], @@ -41,25 +25,9 @@ var Suggestion = Widget.extend({ }, init: function (parent, options) { - this.root = options.root; this.keyword = options.keyword; - this.language = options.language; - this.htmlPage = options.page; this._super(parent); }, - start: function () { - this.htmlPage.on('title-changed', this, this.renderElement); - this.htmlPage.on('description-changed', this, this.renderElement); - }, - analyze: function () { - return analyzeKeyword(this.htmlPage, this.keyword); - }, - highlight: function () { - return this.analyze().title; - }, - tooltip: function () { - return this.analyze().description; - }, select: function () { this.trigger('selected', this.keyword); }, @@ -72,7 +40,7 @@ var SuggestionList = Widget.extend({ init: function (parent, options) { this.root = options.root; this.language = options.language; - this.htmlPage = options.page; + this.htmlPage = options.htmlPage; this._super(parent); }, start: function () { @@ -96,7 +64,7 @@ var SuggestionList = Widget.extend({ var self = this; self.$el.empty(); // TODO Improve algorithm + Ajust based on custom user keywords - var regex = new RegExp(self.root, 'gi'); + var regex = new RegExp(WORD_SEPARATORS_REGEX + self.root + WORD_SEPARATORS_REGEX, 'gi'); keywords = _.map(_.uniq(keywords), function (word) { return word.replace(regex, '').trim(); }); @@ -104,10 +72,7 @@ var SuggestionList = Widget.extend({ _.each(keywords, function (keyword) { if (keyword) { var suggestion = new Suggestion(self, { - root: self.root, - language: self.language, keyword: keyword, - page: self.htmlPage, }); suggestion.on('selected', self, function (word, language) { self.trigger('selected', word, language); @@ -124,45 +89,53 @@ var Keyword = Widget.extend({ events: { 'click a[data-action=remove-keyword]': 'destroy', }, - maxWordsPerKeyword: 4, // TODO Check init: function (parent, options) { this.keyword = options.word; this.language = options.language; - this.htmlPage = options.page; + this.htmlPage = options.htmlPage; + this.used_h1 = this.htmlPage.isInHeading1(this.keyword); + this.used_h2 = this.htmlPage.isInHeading2(this.keyword); + this.used_content = this.htmlPage.isInBody(this.keyword); this._super(parent); }, start: function () { - this.htmlPage.on('title-changed', this, this.updateLabel); - this.htmlPage.on('description-changed', this, this.updateLabel); + this.$('.js_seo_keyword_suggestion').empty(); this.suggestionList = new SuggestionList(this, { root: this.keyword, language: this.language, - page: this.htmlPage, + htmlPage: this.htmlPage, }); this.suggestionList.on('selected', this, function (word, language) { this.trigger('selected', word, language); }); this.suggestionList.appendTo(this.$('.js_seo_keyword_suggestion')); - }, - analyze: function () { - return analyzeKeyword(this.htmlPage, this.keyword); - }, - highlight: function () { - return this.analyze().title; - }, - tooltip: function () { - return this.analyze().description; - }, - updateLabel: function () { - var cssClass = 'oe_seo_keyword js_seo_keyword ' + this.highlight(); - this.$('.js_seo_keyword').attr('class', cssClass); - this.$('.js_seo_keyword').attr('title', this.tooltip()); + + this.htmlPage.on('title-changed', this, this._updateTitle); + this.htmlPage.on('description-changed', this, this._updateDescription); + this._updateTitle(); + this._updateDescription(); }, destroy: function () { this.trigger('removed'); this._super(); }, + _updateTitle: function () { + var $title = this.$('.js_seo_keyword_title'); + if (this.htmlPage.isInTitle(this.keyword)) { + $title.css('visibility','visible'); + } else { + $title.css('visibility','hidden'); + } + }, + _updateDescription: function () { + var $description = this.$('.js_seo_keyword_description'); + if (this.htmlPage.isInDescription(this.keyword)) { + $description.css('visibility','visible'); + } else { + $description.css('visibility','hidden'); + } + }, }); var KeywordList = Widget.extend({ @@ -171,7 +144,7 @@ var KeywordList = Widget.extend({ maxKeywords: 10, init: function (parent, options) { - this.htmlPage = options.page; + this.htmlPage = options.htmlPage; this._super(parent); }, start: function () { @@ -204,11 +177,10 @@ var KeywordList = Widget.extend({ var keyword = new Keyword(self, { word: word, language: language, - page: this.htmlPage, + htmlPage: this.htmlPage, }); keyword.on('removed', self, function () { self.trigger('list-not-full'); - self.trigger('removed', word); self.trigger('content-updated', true); }); keyword.on('selected', self, function (word, language) { @@ -230,15 +202,23 @@ var Preview = Widget.extend({ init: function (parent, options) { this.title = options.title; this.url = options.url; - this.description = options.description || "[ The description will be generated by google unless you specify one ]"; + this.description = options.description || _t("The description will be generated by search engines based on page content unless you specify one."); + if (this.description.length > 160) { + this.description = this.description.substring(0,159) + '…'; + } this._super(parent); }, }); var HtmlPage = Class.extend(mixins.PropertiesMixin, { + init: function () { + mixins.PropertiesMixin.init.call(this); + this.initTitle = this.title(); + this.initDescription = this.description(); + }, url: function () { var url = window.location.href; - var hashIndex = url.indexOf('#'); + var hashIndex = url.indexOf('?'); return hashIndex >= 0 ? url.substring(0, hashIndex) : url; }, title: function () { @@ -247,7 +227,7 @@ var HtmlPage = Class.extend(mixins.PropertiesMixin, { }, changeTitle: function (title) { // TODO create tag if missing - $('title').text(title); + $('title').text(title.trim() || this.initTitle); this.trigger('title-changed', title); }, description: function () { @@ -267,7 +247,6 @@ var HtmlPage = Class.extend(mixins.PropertiesMixin, { changeKeywords: function (keywords) { // TODO create tag if missing $('meta[name=keywords]').attr('content', keywords.join(',')); - this.trigger('keywords-changed', keywords); }, headers: function (tag) { return $('#wrap '+tag).map(function () { @@ -277,9 +256,11 @@ var HtmlPage = Class.extend(mixins.PropertiesMixin, { getOgMeta: function () { var ogImageUrl = $('meta[property="og:image"]').attr('content'); var title = $('meta[property="og:title"]').attr('content'); + var description = $('meta[property="og:description"]').attr('content'); return { ogImageUrl: ogImageUrl && ogImageUrl.replace(window.location.origin, ''), metaTitle: title, + metaDescription: description, }; }, images: function () { @@ -295,7 +276,13 @@ var HtmlPage = Class.extend(mixins.PropertiesMixin, { return $('html').attr('data-oe-company-name'); }, bodyText: function () { - return $('body').children().not('.js_seo_configuration').text(); + return $('body').children().not('.oe_seo_configuration').text(); + }, + heading1: function () { + return $('body').children().not('.oe_seo_configuration').find('h1').text(); + }, + heading2: function () { + return $('body').children().not('.oe_seo_configuration').find('h2').text(); }, isInBody: function (text) { return new RegExp(WORD_SEPARATORS_REGEX+text+WORD_SEPARATORS_REGEX, 'gi').test(this.bodyText()); @@ -306,25 +293,222 @@ var HtmlPage = Class.extend(mixins.PropertiesMixin, { isInDescription: function (text) { return new RegExp(WORD_SEPARATORS_REGEX+text+WORD_SEPARATORS_REGEX, 'gi').test(this.description()); }, + isInHeading1: function (text) { + return new RegExp(WORD_SEPARATORS_REGEX+text+WORD_SEPARATORS_REGEX, 'gi').test(this.heading1()); + }, + isInHeading2: function (text) { + return new RegExp(WORD_SEPARATORS_REGEX+text+WORD_SEPARATORS_REGEX, 'gi').test(this.heading2()); + }, }); -var Tip = Widget.extend({ - template: 'website.seo_tip', +var MetaTitleDescription = Widget.extend({ + // Form and preview for SEO meta title and meta description + // + // We only want to show an alert for "description too small" on those cases + // - at init and the description is not empty + // - we reached past the minimum and went back to it + // - focus out of the field + // Basically we don't want the too small alert when the field is empty and + // we start typing on it. + template: 'website.seo_meta_title_description', xmlDependencies: ['/website/static/src/xml/website.seo.xml'], events: { - 'closed.bs.alert': 'destroy', + 'input input[name=website_meta_title]': '_titleChanged', + 'input textarea[name=website_meta_description]': '_descriptionOnInput', + 'change textarea[name=website_meta_description]': '_descriptionOnChange', }, + maxRecommendedDescriptionSize: 300, + minRecommendedDescriptionSize: 50, + showDescriptionTooSmall: false, + /** + * @override + */ init: function (parent, options) { - this.message = options.message; - // cf. http://getbootstrap.com/components/#alerts - // success, info, warning or danger - this.type = options.type || 'info'; - this._super(parent); + this.htmlPage = options.htmlPage; + this.canEditTitle = !!options.canEditTitle; + this.canEditDescription = !!options.canEditDescription; + this.isIndexed = !!options.isIndexed; + this._super(parent, options); + }, + /** + * @override + */ + start: function () { + this.$title = this.$('input[name=website_meta_title]'); + this.$description = this.$('textarea[name=website_meta_description]'); + this.$warning = this.$('div#website_meta_description_warning'); + this.$preview = this.$('.js_seo_preview'); + + this._renderPreview(); + + if (!this.canEditTitle) { + this.$title.attr('disabled', true); + } + if (!this.canEditDescription) { + this.$description.attr('disabled', true); + } + + this.$title.val(this.htmlPage.title()); + this.$description.val(this.htmlPage.description()); + + this._descriptionOnChange(); + }, + /** + * Get the current title + */ + getTitle: function () { + return this.$title.val().trim() || this.htmlPage.initTitle; + }, + /** + * Get the current description + */ + getDescription: function () { + return this.$description.val(); + }, + /** + * @private + */ + _titleChanged: function () { + var self = this; + self._renderPreview(); + self.trigger('title-changed'); + }, + /** + * @private + */ + _descriptionOnChange: function () { + this.showDescriptionTooSmall = true; + this._descriptionOnInput(); + }, + /** + * @private + */ + _descriptionOnInput: function () { + var length = this.getDescription().length; + + if (length >= this.minRecommendedDescriptionSize) { + this.showDescriptionTooSmall = true; + } else if (length === 0) { + this.showDescriptionTooSmall = false; + } + + if (length > this.maxRecommendedDescriptionSize) { + this.$warning.text(_t('Your description looks too long.')).show(); + } else if (this.showDescriptionTooSmall && length < this.minRecommendedDescriptionSize) { + this.$warning.text(_t('Your description looks too short.')).show(); + } else { + this.$warning.hide(); + } + + this._renderPreview(); + this.trigger('description-changed'); + }, + /** + * @private + */ + _renderPreview: function () { + var indexed = this.isIndexed; + var preview = ""; + if (indexed){ + preview = new Preview(this, { + title: this.getTitle(), + description: this.getDescription(), + url: this.htmlPage.url(), + }); + } else { + preview = new Preview(this, { + description: _t("You have hidden this page from search results. It won't be indexed by search engines."), + }); + } + this.$preview.empty(); + preview.appendTo(this.$preview); + }, +}); + +var MetaKeywords = Widget.extend({ + // Form and table for SEO meta keywords + template: 'website.seo_meta_keywords', + xmlDependencies: ['/website/static/src/xml/website.seo.xml'], + events: { + 'keyup input[name=website_meta_keywords]': '_confirmKeyword', + 'click button[data-action=add]': '_addKeyword', + }, + + init: function (parent, options) { + this.htmlPage = options.htmlPage; + this._super(parent, options); + }, + start: function () { + var self = this; + this.$input = this.$('input[name=website_meta_keywords]'); + this.keywordList = new KeywordList(this, { htmlPage: this.htmlPage }); + this.keywordList.on('list-full', this, function () { + self.$input.attr({ + readonly: 'readonly', + placeholder: "Remove a keyword first" + }); + self.$('button[data-action=add]').prop('disabled', true).addClass('disabled'); + }); + this.keywordList.on('list-not-full', this, function () { + self.$input.removeAttr('readonly').attr('placeholder', ""); + self.$('button[data-action=add]').prop('disabled', false).removeClass('disabled'); + }); + this.keywordList.on('selected', this, function (word, language) { + self.keywordList.add(word, language); + }); + this.keywordList.on('content-updated', this, function (removed) { + self._updateTable(removed); + }); + this.keywordList.insertAfter(this.$('.table thead')); + + this._getLanguages(); + this._updateTable(); + }, + _addKeyword: function () { + var $language = this.$('select[name=seo_page_language]'); + var keyword = this.$input.val(); + var language = $language.val().toLowerCase(); + this.keywordList.add(keyword, language); + this.$input.val('').focus(); + }, + _confirmKeyword: function (e) { + if (e.keyCode === 13) { + this._addKeyword(); + } + }, + _getLanguages: function () { + var self = this; + this._rpc({ + model: 'website', + method: 'get_languages', + args: [[weContext.get().website_id]], + context: weContext.get(), + }).then( function (data) { + self.$('#language-box').html(core.qweb.render('Configurator.language_promote', { + 'language': data, + 'def_lang': weContext.get().lang + })); + }); + }, + /* + * Show the table if there is at least one keyword. Hide it otherwise. + * + * @private + * @param {boolean} removed: a keyword is about to be removed, + * we need to exclude it from the count + */ + _updateTable : function (removed) { + var min = removed ? 1 : 0; + if (this.keywordList.keywords().length > min) { + this.$('table').show(); + } else { + this.$('table').hide(); + } }, }); -var metaImageSelector = Widget.extend({ +var MetaImageSelector = Widget.extend({ template: 'website.seo_meta_image_selector', xmlDependencies: ['/website/static/src/xml/website.seo.xml'], events: { @@ -337,19 +521,40 @@ var metaImageSelector = Widget.extend({ * @param {Object} data */ init: function (parent, data) { - this.metaTitle = data.title; + this.metaTitle = data.title || ''; + this._setDescription(data.description); this.activeMetaImg = data.metaImg; - this.serverUrl = window.location.origin; + this.serverUrl = data.htmlpage.url(); data.pageImages.unshift(_.str.sprintf('/web/image/res.company/%s/logo', odoo.session_info.website_company_id)); this.images = _.uniq(data.pageImages); this.customImgUrl = _.contains(data.pageImages, data.metaImg) ? false : data.metaImg; this._super(parent); }, + setTitle: function (title) { + this.metaTitle = title; + this._updateTemplateBody(); + }, + setDescription: function (description) { + this._setDescription(description); + this._updateTemplateBody(); + }, //-------------------------------------------------------------------------- // Private //-------------------------------------------------------------------------- + /** + * Set the description, applying ellipsis if too long. + * + * @private + */ + _setDescription: function (description) { + this.metaDescription = description || _t("The description will be generated by social media based on page content unless you specify one."); + if (this.metaDescription.length > 160) { + this.metaDescription = this.metaDescription.substring(0,159) + '…'; + } + }, + /** * Update template. * @@ -404,24 +609,15 @@ var SeoConfigurator = Dialog.extend({ xmlDependencies: Dialog.prototype.xmlDependencies.concat( ['/website/static/src/xml/website.seo.xml'] ), - events: { - 'keyup input[name=seo_page_keywords]': 'confirmKeyword', - 'blur input[name=seo_page_title]': 'titleChanged', - 'blur textarea[name=seo_page_description]': 'descriptionChanged', - 'click button[data-action=add]': 'addKeyword', - }, canEditTitle: false, canEditDescription: false, canEditKeywords: false, canEditLanguage: false, - maxTitleSize: 65, - maxDescriptionSize: 160, // TODO master: remove me and add warning init: function (parent, options) { options = options || {}; _.defaults(options, { - title: _t('Promote Page'), - subtitle: _t('Get this page efficiently referenced in search engines to attract more visitors.'), + title: _t('Optimize SEO'), buttons: [ {text: _t('Save'), classes: 'btn-primary', click: this.update}, {text: _t('Discard'), close: true}, @@ -433,106 +629,71 @@ var SeoConfigurator = Dialog.extend({ start: function () { var self = this; - this.$modal.addClass('oe_seo_configuration js_seo_configuration'); + this.$modal.addClass('oe_seo_configuration'); this.htmlPage = new HtmlPage(); - this.$('.js_seo_page_url').text(this.htmlPage.url()); - this.$('input[name=seo_page_title]').val(this.htmlPage.title()); - this.$('textarea[name=seo_page_description]').val(this.htmlPage.description()); - this.keywordList = new KeywordList(self, { page: this.htmlPage }); - this.keywordList.on('list-full', self, function () { - self.$('input[name=seo_page_keywords]').attr({ - readonly: 'readonly', - placeholder: "Remove a keyword first" - }); - self.$('button[data-action=add]').prop('disabled', true).addClass('disabled'); - }); - this.keywordList.on('list-not-full', self, function () { - self.$('input[name=seo_page_keywords]').removeAttr('readonly').attr('placeholder', ""); - self.$('button[data-action=add]').prop('disabled', false).removeClass('disabled'); - }); - this.keywordList.on('selected', self, function (word, language) { - self.keywordList.add(word, language); - }); - this.keywordList.on('content-updated', self, function (removed) { - self.updateTable(removed); - }); - this.keywordList.insertAfter(this.$('.table thead')); - this.disableUnsavableFields().then(function(){ - self.renderPreview(); - self.metaImageSelector = new metaImageSelector(self, { + this.disableUnsavableFields().then(function () { + // Image selector + self.metaImageSelector = new MetaImageSelector(self, { + htmlpage: self.htmlPage, title: self.htmlPage.getOgMeta().metaTitle, + description: self.htmlPage.getOgMeta().metaDescription, metaImg : self.metaImg || self.htmlPage.getOgMeta().ogImageUrl, pageImages : _.pluck(self.htmlPage.images().get(), 'src'), }); - self.metaImageSelector.appendTo(self.$('.js_seo_keywords_list')); - }); + self.metaImageSelector.appendTo(self.$('.js_seo_image')); + + // title and description + self.metaTitleDescription = new MetaTitleDescription(self, { + htmlPage: self.htmlPage, + canEditTitle: self.canEditTitle, + canEditDescription: self.canEditDescription, + isIndexed: self.isIndexed, + }); + self.metaTitleDescription.on('title-changed', self, self.titleChanged); + self.metaTitleDescription.on('description-changed', self, self.descriptionChanged); + self.metaTitleDescription.appendTo(self.$('.js_seo_meta_title_description')); - this.getLanguages(); - this.updateTable(); - }, - getLanguages: function () { - var self = this; - this._rpc({ - model: 'website', - method: 'get_languages', - args: [[weContext.get().website_id]], - context: weContext.get(), - }).then( function (data) { - self.$('#language-box').html(core.qweb.render('Configurator.language_promote', { - 'language': data, - 'def_lang': weContext.get().lang - })); + // keywords + self.metaKeywords = new MetaKeywords(self, {htmlPage: self.htmlPage}); + self.metaKeywords.appendTo(self.$('.js_seo_meta_keywords')); }); }, + /* + * Reset meta tags to their initial value if not saved. + * + * @private + */ + destroy: function () { + if (!this.savedData) { + this.htmlPage.changeTitle(this.htmlPage.initTitle); + this.htmlPage.changeDescription(this.htmlPage.initDescription); + } + this._super.apply(this, arguments); + }, disableUnsavableFields: function () { var self = this; return this.loadMetaData().then(function (data) { + // We only need a reload for COW when the copy is happening, therefore: + // - no reload if we are not editing a view (condition: website_id === undefined) + // - reload if generic page (condition: website_id === false) + self.reloadOnSave = data.website_id === undefined ? false : !data.website_id; //If website.page, hide the google preview & tell user his page is currently unindexed self.isIndexed = (data && ('website_indexed' in data)) ? data.website_indexed : true; self.canEditTitle = data && ('website_meta_title' in data); self.canEditDescription = data && ('website_meta_description' in data); self.canEditKeywords = data && ('website_meta_keywords' in data); self.metaImg = data.website_meta_og_img; - if (!self.canEditTitle) { - self.$('input[name=seo_page_title]').attr('disabled', true); - } - if (!self.canEditDescription) { - self.$('textarea[name=seo_page_description]').attr('disabled', true); - } if (!self.canEditTitle && !self.canEditDescription && !self.canEditKeywords) { + // disable the button to prevent an error if the current page doesn't use the mixin + // we make the check here instead of on the view because we don't need to check + // at every page load, just when the rare case someone clicks on this link + // TODO don't show the modal but just an alert in this case self.$footer.find('button[data-action=update]').attr('disabled', true); } }); }, - suggestImprovements: function () { - var self = this; - var tips = []; - _.each(tips, function (tip) { - displayTip(tip.message, tip.type); - }); - - function displayTip(message, type) { - new Tip(self, { - message: message, - type: type, - }).appendTo(self.$('.js_seo_tips')); - } - }, - confirmKeyword: function (e) { - if (e.keyCode === 13) { - this.addKeyword(); - } - }, - addKeyword: function (word) { - var $input = this.$('input[name=seo_page_keywords]'); - var $language = this.$('select[name=seo_page_language]'); - var keyword = _.isString(word) ? word : $input.val(); - var language = $language.val().toLowerCase(); - this.keywordList.add(keyword, language); - $input.val('').focus(); - }, update: function () { var self = this; var data = {}; @@ -543,12 +704,20 @@ var SeoConfigurator = Dialog.extend({ data.website_meta_description = this.htmlPage.description(); } if (this.canEditKeywords) { - data.website_meta_keywords = this.keywordList.keywords().join(', '); + data.website_meta_keywords = this.metaKeywords.keywordList.keywords().join(', '); } data.website_meta_og_img = this.metaImageSelector.activeMetaImg; this.saveMetaData(data).then(function () { - self.htmlPage.changeKeywords(self.keywordList.keywords()); - self.close(); + // We want to reload if we are editing a generic page + // because it will become a specific page after this change (COW) + // and we want the user to be on the page he just created. + if (self.reloadOnSave) { + window.location.href = self.htmlPage.url(); + } else { + self.htmlPage.changeKeywords(self.metaKeywords.keywordList.keywords()); + self.savedData = true; + self.close(); + } }); }, getMainObject: function () { @@ -572,8 +741,9 @@ var SeoConfigurator = Dialog.extend({ } else { var fields = ['website_meta_title', 'website_meta_description', 'website_meta_keywords' ,'website_meta_og_img']; - if (obj.model == 'website.page'){ + if (obj.model === 'website.page'){ fields.push('website_indexed'); + fields.push('website_id'); } rpc.query({ model: obj.model, @@ -608,44 +778,19 @@ var SeoConfigurator = Dialog.extend({ titleChanged: function () { var self = this; _.defer(function () { - var title = self.$('input[name=seo_page_title]').val(); + var title = self.metaTitleDescription.getTitle(); self.htmlPage.changeTitle(title); - self.renderPreview(); + self.metaImageSelector.setTitle(title); }); }, descriptionChanged: function () { var self = this; _.defer(function () { - var description = self.$('textarea[name=seo_page_description]').val(); + var description = self.metaTitleDescription.getDescription(); self.htmlPage.changeDescription(description); - self.renderPreview(); + self.metaImageSelector.setDescription(description); }); }, - renderPreview: function () { - var indexed = this.isIndexed; - var preview = ""; - if(indexed){ - preview = new Preview(this, { - title: this.htmlPage.title(), - description: this.htmlPage.description(), - url: this.htmlPage.url(), - }); - } - else{ - preview = new Preview(this, { - description: _t("You have hidden this page from search results. It won't be indexed by search engines."), - }); - } - var $preview = this.$('.js_seo_preview'); - $preview.empty(); - preview.appendTo($preview); - }, - updateTable : function (removed) { - var self = this, - val = removed ? (this.$el.find('tbody > tr').length - 1) : (this.$el.find('tbody > tr').length); - this.$('table').toggleClass('js_seo_has_content', val > 0 ); - this.$el.scrollTop(self.$el[0].scrollHeight); - }, }); var SeoMenu = websiteNavbarData.WebsiteNavbarActionWidget.extend({ @@ -653,6 +798,14 @@ var SeoMenu = websiteNavbarData.WebsiteNavbarActionWidget.extend({ 'promote-current-page': '_promoteCurrentPage', }), + init: function (parent, options) { + this._super(parent, options); + + if (window.location.href.includes('enable_seo')) { + this._promoteCurrentPage(); + } + }, + //-------------------------------------------------------------------------- // Actions //-------------------------------------------------------------------------- diff --git a/addons/website/static/src/scss/website.ui.scss b/addons/website/static/src/scss/website.ui.scss index 19532adcc79d..6cf572925c3a 100644 --- a/addons/website/static/src/scss/website.ui.scss +++ b/addons/website/static/src/scss/website.ui.scss @@ -193,7 +193,6 @@ body .modal { #language-box { padding-right: 25px; background-color: white; - margin-left: -1px; } .o_seo_og_image { .o_meta_img { @@ -259,121 +258,7 @@ body .modal { } } - .table { - td { - vertical-align: middle; - - &:first-child { - padding-right: 15px; - border-width: 0; - width: 35%; - } - - &:last-child { - visibility: hidden; - } - } - - > tfoot { - display: none; - } - - &.js_seo_has_content { - td { - &:first-child { - width: 55%; - padding-right: 15px; - border: 1px solid $o-we-color-text-light; - text-align: right; - background-color: $o-we-color-paper; - - @include media-breakpoint-up(lg) { - width: 38%; - } - } - - &:last-child { - border: none; - visibility: visible; - padding-left: 15px; - - @include media-breakpoint-up(lg) { - padding-left: 50px; - } - } - } - - tbody { - td { - padding: 5px; - transition: padding .3s ease 0s; - - &:first-child { - border-width: 0 1px; - - .label { - position: relative; - display: inline-block; - padding: 7px 35px; - font-size: 16px; - font-weight: normal; - - .oe_remove { - @include o-w-close-icon(10px, white, $o-we-color-danger, 2px); // FIXME - @include o-position-absolute(5px, 5px); - } - } - } - - &:last-child { - .label { - display: block; - font-size: 12px; - font-weight: normal; - opacity: 0.8; - cursor: pointer; - - &:hover { - opacity: 1; - } - } - } - } - - tr { - animation: fadeInDownSmall .3s ease 0s 1 normal none running; - - &:first-child { - td:first-child { - padding-top: 10px; - } - } - - &:last-child { - td:first-child { - padding-bottom: 10px; - border-bottom-width: 1px; - } - } - } - } - - > tfoot { - display: table-footer-group; - - hr { - margin: 10px 0; - } - - td, td:first-child { - border: none; - background: none; - } - } - } - } - - li.oe_seo_preview_g { + div.oe_seo_preview_g { list-style: none; font-family: arial, sans-serif; @@ -396,9 +281,6 @@ body .modal { font-size: 14px; line-height: 18px; } - .st { - height: 50px; - } } } } diff --git a/addons/website/static/src/xml/website.seo.xml b/addons/website/static/src/xml/website.seo.xml index 8ada1a3088a7..e6bb6aa378a7 100644 --- a/addons/website/static/src/xml/website.seo.xml +++ b/addons/website/static/src/xml/website.seo.xml @@ -6,72 +6,10 @@ </t> </t> - <div t-name="website.seo_configuration"> - <section class="row"> - <div class="col-lg-6"> - <div class="mt16" role="form"> - <div class="form-group row"> - <label for="seo_page_title" class="col-3 col-form-label">Page Title</label> - <div class="col-9"> - <input type="text" name="seo_page_title" class="form-control" maxlength="70" size="70"/> - </div> - </div> - <div class="form-group row"> - <label for="seo_page_description" class="col-3 col-form-label">Description</label> - <div class="col-9"> - <textarea name="seo_page_description" class="form-control" style="max-width: 100%;height:145px;max-height:145px" t-att-maxlength="widget.maxDescriptionSize"/> - </div> - </div> - </div> - </div> - <div class="col-lg-6"> - <h5 class="mt16">Preview <small>how your page will be listed on Google</small></h5> - <div> - <div class="card mb0"> - <div class="card-body"> - <div class="js_seo_preview"/> - </div> - </div> - </div> - </div> - </section> - <hr/> - <div class="js_seo_tips" /> - <section class="js_seo_keywords_list"> - <h4 class="mt16">Define Keywords <small>describing your page content</small></h4> - <table class="table table-responsive"> - <thead> - <tr> - <td> - <div class="form-inline" role="form"> - <div class="input-group"> - <input type="text" style="min-width:170px" name="seo_page_keywords" class="form-control" placeholder="Keyword" maxlength="30"/> - <span class="input-group-append"> - <select name="seo_page_language" id="language-box" class="btn form-control"/> - </span> - <span class="input-group-append"> - <button data-action="add" class="btn btn-success" type="button">Add</button> - </span> - </div> - </div> - </td> - <td>Suggested<br/><small class="text-muted">Most searched topics related to your keywords, ordered by importance:</small></td> - </tr> - </thead> - <tfoot> - <tr> - <td/> - <td><hr/> - <small class="text-muted">Legend:</small> - <span class="badge badge-secondary">Not used</span> - <span class="badge badge-success">In title</span> - <span class="badge badge-primary">In description</span> - <span class="badge badge-info">In page's content</span> - </td> - </tr> - </tfoot> - </table> - </section> + <div t-name="website.seo_configuration" role="form"> + <section class="js_seo_meta_title_description"/> + <section class="js_seo_meta_keywords"/> + <section class="js_seo_image"/> </div> <t t-name="website.seo_suggestion_list"> @@ -86,35 +24,27 @@ </tbody> </t> - <t t-name="website.seo_tip"> - <div t-attf-class="alert alert-#{ widget.type }" role="alert"> - <t t-raw="widget.message"/> - </div> - </t> - <t t-name="website.seo_keyword"> - <tr> - <td> - <span t-attf-title="#{ widget.tooltip() }" t-attf-class="oe_seo_keyword #{ widget.highlight() } js_seo_keyword" t-att-data-keyword="widget.keyword"> - <t t-raw="widget.keyword"/> <a href="#" class="oe_remove" data-action="remove-keyword"/> - </span> - </td> - <td class="js_seo_keyword_suggestion"> - <!-- filled in JS --> - </td> + <tr class="js_seo_keyword" t-att-data-keyword="widget.keyword"> + <td t-esc="widget.keyword"/> + <td class="text-center"><i t-if="widget.used_h1" class="fa fa-check" t-attf-title="{{ widget.keyword }} is used in page first level heading"/></td> + <td class="text-center"><i t-if="widget.used_h2" class="fa fa-check" t-attf-title="{{ widget.keyword }} is used in page second level heading"/></td> + <td class="text-center"><i class="js_seo_keyword_title fa fa-check" style="visibility: hidden;" t-attf-title="{{ widget.keyword }} is used in page title"/></td> + <td class="text-center"><i class="js_seo_keyword_description fa fa-check" style="visibility: hidden;" t-attf-title="{{ widget.keyword }} is used in page description"/></td> + <td class="text-center"><i t-if="widget.used_content" class="fa fa-check" t-attf-title="{{ widget.keyword }} is used in page content"/></td> + <td class="js_seo_keyword_suggestion"/> + <td class="text-center"><a href="#" class="oe_remove" data-action="remove-keyword" t-attf-title="Remove {{ widget.keyword }}"><i class="fa fa-trash"/></a></td> </tr> </t> <t t-name="website.seo_suggestion"> - <li class="list-inline-item oe_seo_suggestion"> - <span t-attf-title="#{ widget.tooltip() }" t-attf-class="oe_seo_keyword #{ widget.highlight() } js_seo_suggestion" t-att-data-keyword="widget.keyword"> - <t t-raw="widget.keyword"/> - </span> + <li class="list-inline-item"> + <span class="js_seo_suggestion badge badge-info" t-att-data-keyword="widget.keyword" t-esc="widget.keyword"/> </li> </t> <t t-name="website.seo_preview"> - <li class="oe_seo_preview_g"> + <div class="oe_seo_preview_g"> <div class="rc"> <div class="r"><t t-esc="widget.title"/></div> <div class="s"> @@ -122,7 +52,70 @@ <div class="st"><t t-esc="widget.description"/></div> </div> </div> - </li> + </div> + </t> + + <div t-name="website.seo_meta_title_description"> + <h4><small>Optimize this page's referencing in search engines</small></h4> + <div class="row"> + <div class="col-lg-6"> + <div class="form-group"> + <label for="website_meta_title"> + Title <i class="fa fa-question-circle-o" title="The title will take a default value unless you specify one."/> + </label> + <input type="text" name="website_meta_title" id="website_meta_title" class="form-control" maxlength="70" size="70"/> + </div> + <div class="form-group"> + <label for="website_meta_description"> + Description <i class="fa fa-question-circle-o" title="The description will be generated by search engines based on page content unless you specify one."/> + </label> + <textarea name="website_meta_description" id="website_meta_description" class="form-control" style="height:120px;"/> + <div class="alert alert-warning mt16 mb0 small" id="website_meta_description_warning" style="display: none;"/> + </div> + </div> + <div class="col-lg-6"> + <div class="card-header">Preview</div> + <div class="card mb0 p-0"> + <div class="card-body"> + <div class="js_seo_preview"/> + </div> + </div> + </div> + </div> + </div> + + <t t-name="website.seo_meta_keywords"> + <label for="website_meta_keywords"> + Keywords + </label> + <div class="form-inline" role="form"> + <div class="input-group"> + <input type="text" name="website_meta_keywords" id="website_meta_keywords" class="form-control" placeholder="Keyword" maxlength="30"/> + <span class="input-group-append"> + <select name="seo_page_language" id="language-box" class="btn form-control"/> + </span> + <span class="input-group-append"> + <button data-action="add" class="btn btn-success" type="button">Add</button> + </span> + </div> + </div> + <div class="table-responsive mt16"> + <table class="table"> + <thead> + <tr> + <th>Keyword</th> + <th class="text-center" title="Used in page first level heading">H1</th> + <th class="text-center" title="Used in page second level heading">H2</th> + <th class="text-center" title="Used in page title">T</th> + <th class="text-center" title="Used in page description">D</th> + <th class="text-center" title="Used in page content">C</th> + <th title="Most searched topics related to your keyword, ordered by importance">Related keywords</th> + <th class="text-center"><i class="fa fa-trash" title="Remove"/></th> + </tr> + </thead> + <!-- body inserted in JS --> + </table> + </div> </t> <div t-name="website.seo_meta_image_selector" class="o_seo_og_image"> @@ -130,10 +123,9 @@ </div> <t t-name="website.og_image_body"> + <h4><small>Select an image for social share</small></h4> <div class="row"> - <div class="col-md-6"> - <h5 class="mt16 mb4">Image for Social Share</h5> - <div class="text-muted mb32">Select an image to use in social share.</div> + <div class="col-lg-6"> <t t-foreach="widget.images" t-as="image"> <div t-attf-class="o_meta_img mt4 #{image === widget.activeMetaImg and ' o_active_image' or ''}"> <img t-att-src="image"/> @@ -147,13 +139,14 @@ <i class="fa fa-upload"/> </div> </div> - <div class="col-md-6"> - <h6 class="mt16">Preview <small>how your page will be displayed on social media</small></h6> - <div class="card p-0"> + <div class="col-lg-6"> + <div class="card p-0 mb16"> + <div class="card-header">Preview</div> <img class="card-img-top o_meta_active_img" t-att-src="widget.activeMetaImg"/> <div class="card-body px-3 py-2"> <h6 class="text-alpha card-title mb0"><t t-esc="widget.metaTitle"/></h6> <small class="card-subtitle text-muted"><t t-esc="widget.serverUrl"/></small> + <p t-esc="widget.metaDescription"/> </div> </div> </div> diff --git a/addons/website/views/website_templates.xml b/addons/website/views/website_templates.xml index 568e44105349..73a290eb19e2 100644 --- a/addons/website/views/website_templates.xml +++ b/addons/website/views/website_templates.xml @@ -132,13 +132,13 @@ 'data-oe-company-name': res_company.name }"/> <t t-if="not title"> - <t t-if="not additional_title and main_object and 'name' in main_object"> - <t t-set="additional_title" t-value="main_object.name"/> - </t> <t t-if="main_object and 'website_meta_title' in main_object and main_object.website_meta_title"> <t t-set="title" t-value="main_object.website_meta_title"/> </t> <t t-else=""> + <t t-if="not additional_title and main_object and 'name' in main_object"> + <t t-set="additional_title" t-value="main_object.name"/> + </t> <t t-set="title"><t t-if="additional_title"><t t-raw="additional_title"/> | </t><t t-raw="(website or res_company).name"/></t> </t> </t> @@ -1195,7 +1195,8 @@ Sitemap: <t t-esc="url_root"/>sitemap.xml <th class="text-center"><i title="Is the page included in the main menu?" class="fa fa-thumb-tack"></i></th> <th class="text-center"><i title="Is the page published?" class="fa fa-eye"></i></th> <th class="text-center"><i title="Is the page indexed by search engines?" class="fa fa-globe"></i></th> - <th style='width:140px;'></th> + <th class="text-center"><i title="Is the page SEO optimized?" class="fa fa-search"></i></th> + <th></th> </tr> </thead> <t t-set='prev_page' t-value='False' /> @@ -1219,31 +1220,45 @@ Sitemap: <t t-esc="url_root"/>sitemap.xml </template> <template id="one_page_line"> - <t t-set='specific_page' t-value="page.website_id" /> - <t t-set='final_page' t-value="(next_page and page.url != next_page.url) or not next_page or specific_page"/> - <tr t-att-style='not final_page and "color:#999"'> - <td> + <t t-set='specific_page' t-value="page.website_id"/> + <t t-set='final_page' t-value="(next_page and page.url != next_page.url) or not next_page or specific_page"/> + <tr t-att-style='not final_page and "color:#999"'> + <td> <t groups="website.group_multi_website"> - <t t-if='specific_page and prev_page and prev_page.url == page.url and not prev_page.website_id'><i class="fa fa-level-up fa-rotate-90 ml32 mr4"></i></t> - <i t-else="1" class="fa fa-globe mr4" t-att-style='specific_page and "visibility:hidden;"'/> + <i t-if='specific_page and prev_page and prev_page.url == page.url and not prev_page.website_id' class="fa fa-level-up fa-rotate-90 ml32 mr4"/> + <i t-else="1" class="fa fa-globe mr4" t-att-style="'visibility:hidden;' if specific_page else ''"/> </t> - <i t-if="page.is_homepage" class="fa fa-home" title="Home"></i> <span t-esc="page.name"/> - </td> - <td><a t-if='final_page' t-attf-href="{{page.url}}"><t t-esc="page.url"/></a></td> - <td class="text-center"><i t-att-class="'fa fa-check' if page.menu_ids else 'fa fa-times text-muted'" t-att-title="'Checked' if page.menu_ids else 'Not checked'"></i></td> - <t t-set='date_formatted'><t t-options='{"widget": "date"}' t-esc="page.date_publish"/></t> - <td class="text-center"> - <i t-att-title="not page.is_visible and page.website_published and 'This page will be visible on ' + date_formatted" - t-att-class="'fa fa-check' if page.is_visible and page.website_published else 'fa fa-eye-slash' if not page.is_visible and page.website_published else 'fa fa-times text-muted'"></i> - </td> - <td class="text-center"><i t-att-class="'fa fa-check' if page.website_indexed else 'fa fa-times text-muted'" t-att-title="'Checked' if page.website_indexed else 'Not checked'"></i></td> - <td class="text-right" style="white-space:nowrap;"> - <a class="mr4 fa fa-cog js_page_properties" t-att-data-id="page.id" href="#" title="Manage this page"></a> - <a class="mr4 fa fa-pencil-square-o" t-attf-href="/web#id=#{page.view_id.id}&view_type=form&model=ir.ui.view" title="Edit code in backend"></a> - <a class="mr4 fa fa-clone js_clone_page" t-att-data-id="page.id" href="#" title="Clone this page"></a> - <a class="fa fa-trash js_delete_page" t-att-data-id="page.id" href="#" title="Delete this page"></a> - </td> - </tr> + <i t-if="page.is_homepage" class="fa fa-home" title="Home"/> <span t-esc="page.name"/> + </td> + <td> + <a t-if='final_page' t-att-href="page.url"><t t-esc="page.url"/></a> + </td> + <td class="text-center"> + <i t-if="page.menu_ids" class="fa fa-check" title="In main menu"/> + <i t-else="" class="fa fa-times text-muted" title="Not in main menu"/> + </td> + <td class="text-center"> + <t t-set='date_formatted'><t t-options='{"widget": "date"}' t-esc="page.date_publish"/></t> + <i t-if="page.is_visible" class="fa fa-check" title="Visible"/> + <i t-elif="page.website_published" class="fa fa-eye-slash" t-attf-title="This page will be visible on {{ date_formatted }}"/> + <i t-else="" class="fa fa-times text-muted" title="Not visible"/> + </td> + <td class="text-center"> + <i t-if="page.website_indexed" class="fa fa-check" title="Indexed"/> + <i t-else="" class="fa fa-times text-muted" title="Not indexed"/> + </td> + <td class="text-center"> + <i t-if="page.is_seo_optimized" class="fa fa-check" title="SEO optimized"/> + <i t-else="" class="fa fa-times text-muted" title="Not SEO optimized"/> + </td> + <td class="text-right" style="white-space:nowrap;"> + <a class="mr4 fa fa-cog js_page_properties" href="#" t-att-data-id="page.id" title="Manage this page"/> + <a class="mr4 fa fa-search" t-attf-href="{{ page.url}}?enable_seo" title="Optimize SEO of this page"/> + <a groups="base.group_no_one" class="mr4 fa fa-bug" t-attf-href="/web#id=#{page.view_id.id}&view_type=form&model=ir.ui.view" title="Edit code in backend"/> + <a class="mr4 fa fa-clone js_clone_page" t-att-data-id="page.id" href="#" title="Clone this page"/> + <a class="fa fa-trash js_delete_page" t-att-data-id="page.id" href="#" title="Delete this page"/> + </td> + </tr> </template> </odoo> -- GitLab