diff --git a/core/client/controllers/post-settings-menu.js b/core/client/controllers/post-settings-menu.js index 52adb3ec15..1ba92705aa 100644 --- a/core/client/controllers/post-settings-menu.js +++ b/core/client/controllers/post-settings-menu.js @@ -4,6 +4,15 @@ import SlugGenerator from 'ghost/models/slug-generator'; import boundOneWay from 'ghost/utils/bound-one-way'; var PostSettingsMenuController = Ember.ObjectController.extend({ + init: function () { + this._super(); + + // when creating a new post we want to observe the title + // to generate the post's slug + if (this.get('isNew')) { + this.addObserver('title', this, 'titleObserver'); + } + }, isStaticPage: function (key, val) { var self = this; @@ -43,12 +52,14 @@ var PostSettingsMenuController = Ember.ObjectController.extend({ slugGenerator = this.get('slugGenerator'), title = this.get('title'); slugGenerator.generateSlug(title).then(function (slug) { - return self.set('slugPlaceholder', slug); + self.set('slugPlaceholder', slug); }); }, titleObserver: function () { - Ember.run.debounce(this, 'generateSlugPlaceholder', 700); - }.observes('title'), + if (this.get('isNew') && this.get('model').changedAttributes().hasOwnProperty('title')) { + Ember.run.debounce(this, 'generateSlugPlaceholder', 700); + } + }, slugPlaceholder: function (key, value) { var slug = this.get('slug'); @@ -74,22 +85,58 @@ var PostSettingsMenuController = Ember.ObjectController.extend({ var slug = this.get('slug'), self = this; - // Ignore unchanged slugs - if (slug === newSlug) { + newSlug = newSlug || slug; + + newSlug = newSlug.trim(); + + // Ignore unchanged slugs or candidate slugs that are empty + if (!newSlug || slug === newSlug) { return; } - this.set('slug', newSlug); + this.get('slugGenerator').generateSlug(newSlug).then(function (serverSlug) { + // If after getting the sanitized and unique slug back from the API + // we end up with a slug that matches the existing slug, abort the change + if (serverSlug === slug) { + return; + } - //Don't save just yet if it's an empty slug on a draft - if (!newSlug && this.get('isDraft')) { - return; - } + // Because the server transforms the candidate slug by stripping + // certain characters and appending a number onto the end of slugs + // to enforce uniqueness, there are cases where we can get back a + // candidate slug that is a duplicate of the original except for + // the trailing incrementor (e.g., this-is-a-slug and this-is-a-slug-2) - this.get('model').save('slug').then(function () { - self.notifications.showSuccess('Permalink successfully changed to ' + - self.get('slug') + '.'); - }, this.notifications.showErrors); + // get the last token out of the slug candidate and see if it's a number + var slugTokens = serverSlug.split('-'), + check = Number(slugTokens.pop()); + + // if the candidate slug is the same as the existing slug except + // for the incrementor then the existing slug should be used + if (Number.isInteger(check) && check > 0) { + if (slug === slugTokens.join('-') && serverSlug !== newSlug) { + return; + } + } + + self.set('slug', serverSlug); + + if (self.hasObserverFor('title')) { + self.removeObserver('title', this, 'titleObserver'); + } + + // If this is a new post. Don't save the model. Defer the save + // to the user pressing the save button + if (self.get('isNew')) { + return; + } + + // Save post model properties excluding any changes to the post body + return self.get('model').save().then(function () { + self.notifications.showSuccess('Permalink successfully changed to ' + + self.get('slug') + '.'); + }, self.notifications.showErrors); + }); }, /**