0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-01-13 22:41:32 -05:00

Merge pull request #4111 from halfdan/4109-psm

Static page and feature options in new PSM
This commit is contained in:
Hannah Wolfe 2014-09-24 13:54:27 +01:00
commit 7e3d89fb5b
7 changed files with 123 additions and 119 deletions

View file

@ -559,22 +559,6 @@ body.zen {
} }
#entry-controls {
&.unsaved {
.post-settings-menu {
padding-bottom: 0;
.post-setting:nth-child(3) td {
border-bottom: none;
}
.delete {
display: none;
}
}
}
}
#entry-actions { #entry-actions {
margin-right: 6px; margin-right: 6px;
position: relative; position: relative;
@ -589,6 +573,14 @@ body.zen {
bottom: 100%; bottom: 100%;
} }
} }
&.unsaved {
padding-bottom: 0;
.delete {
display: none;
}
}
} }
#entry-actions-menu { #entry-actions-menu {

View file

@ -197,6 +197,22 @@ var PostSettingsMenuController = Ember.ObjectController.extend({
self.get('model').rollback(); self.get('model').rollback();
}); });
}, },
toggleFeatured: function () {
var self = this;
this.toggleProperty('featured');
// If this is a new post. Don't save the model. Defer the save
// to the user pressing the save button
if (this.get('isNew')) {
return;
}
this.get('model').save(this.get('saveOptions')).catch(function (errors) {
self.showErrors(errors);
self.get('model').rollback();
});
},
/** /**
* triggered by user manually changing slug * triggered by user manually changing slug
*/ */

View file

@ -1,13 +1,10 @@
<footer id="publish-bar"> <footer id="publish-bar">
<div class="publish-bar-inner"> <div class="publish-bar-inner">
{{render 'post-tags-input'}} {{render 'post-tags-input'}}
<div class="publish-bar-actions"> <div class="publish-bar-actions">
<button type="button" class='post-settings' {{action "toggleRightOutlet"}}></button> <button type="button" class='post-settings' {{action "toggleRightOutlet"}}></button>
{{view "editor-save-button" id="entry-actions"}} {{view "editor-save-button" id="entry-actions" classNameBindings="isNew:unsaved"}}
</div> </div>
</div> </div>
</footer> </footer>

View file

@ -11,5 +11,9 @@
<li class="post-save-draft" {{bind-attr class="willPublish::active"}}> <li class="post-save-draft" {{bind-attr class="willPublish::active"}}>
<a {{action "setSaveType" "draft"}} href="#">{{view.draftText}}</a> <a {{action "setSaveType" "draft"}} href="#">{{view.draftText}}</a>
</li> </li>
<li class="divider delete"></li>
<li class="delete">
<a {{action "openModal" "delete-post" this}} href="#">Delete Post</a>
</li>
</ul> </ul>
{{/gh-popover}} {{/gh-popover}}

View file

@ -1,6 +1,6 @@
<div class="editor-cover" {{action "closeRightOutlet"}}></div> <div class="editor-cover" {{action "closeRightOutlet"}}></div>
{{#gh-tabs-manager selected="showSubview" id="entry-controls" classNameBindings="isNew:unsaved :right-outlet"}} {{#gh-tabs-manager selected="showSubview" id="entry-controls" class="right-outlet"}}
<div id="entry-controls" {{bind-attr class="isNew:unsaved :right-outlet"}}> <div id="entry-controls" class="right-outlet">
<div {{bind-attr class="isViewingSubview:outlet-pane-out-left:outlet-pane-in :post-settings-menu :outlet-pane"}}> <div {{bind-attr class="isViewingSubview:outlet-pane-out-left:outlet-pane-in :post-settings-menu :outlet-pane"}}>
<div class="post-settings-header"> <div class="post-settings-header">
<h4>Post Settings</h4> <h4>Post Settings</h4>
@ -37,13 +37,6 @@
</span> </span>
</div> </div>
<div class="form-group for-checkbox">
<label class="checkbox" for="static-page" {{action "togglePage" bubbles="false"}}>
{{input type="checkbox" name="static-page" id="static-page" class="post-setting-static-page" checked=page}}
<span class="input-toggle-component"></span>
<p>Static Page</p>
</label>
</div>
<ul class="nav-list nav-list-block"> <ul class="nav-list nav-list-block">
{{#gh-tab tagName="li" classNames="nav-list-item"}} {{#gh-tab tagName="li" classNames="nav-list-item"}}
<a href="#"> <a href="#">
@ -51,28 +44,22 @@
<span>Extra content for SEO and social media.</span> <span>Extra content for SEO and social media.</span>
</a> </a>
{{/gh-tab}} {{/gh-tab}}
{{!--
{{#gh-tab tagName="li" classNames="nav-list-item"}}
<a href="#">
<b>Advanced Settings</b>
<span>Convert to a page, mark as featured.</span>
</a>
{{/gh-tab}}
{{#gh-tab tagName="li" classNames="nav-list-item"}}
<a href="#">
<b>Revision History</b>
<span>12 versions of this post by 3 people.</span>
</a>
{{/gh-tab}}
{{#gh-tab tagName="li" classNames="nav-list-item"}}
<a href="#">
<b>Custom App</b>
<span>Registered an advanced setting panel.</span>
</a>
{{/gh-tab}}
--}}
</ul> </ul>
<button type="button" class="btn btn-red icon-trash delete" {{action "openModal" "delete-post" this}}>Delete This Post</button>
<div class="form-group for-checkbox">
<label class="checkbox" for="static-page" {{action "togglePage" bubbles="false"}}>
{{input type="checkbox" name="static-page" id="static-page" class="post-setting-static-page" checked=page}}
<span class="input-toggle-component"></span>
<p>Turn this post into a static page</p>
</label>
<label class="checkbox" for="static-page" {{action "toggleFeatured" bubbles="false"}}>
{{input type="checkbox" name="featured" id="featured" class="post-setting-featured" checked=featured}}
<span class="input-toggle-component"></span>
<p>Feature this post</p>
</label>
</div>
</form> </form>
</div><!-- .post-settings-content --> </div><!-- .post-settings-content -->
</div><!-- .post-settings-menu --> </div><!-- .post-settings-menu -->

View file

@ -322,7 +322,7 @@ CasperTest.begin('Publish menu - new post', 10, function suite(test) {
}); });
}); });
CasperTest.begin('Publish menu - existing post', 19, function suite(test) { CasperTest.begin('Publish menu - existing post', 23, function suite(test) {
// Create a post, save it and test refreshed editor // Create a post, save it and test refreshed editor
casper.thenOpenAndWaitForPageLoad('editor', function testTitleAndUrl() { casper.thenOpenAndWaitForPageLoad('editor', function testTitleAndUrl() {
test.assertTitle('Ghost Admin', 'Ghost admin has no title'); test.assertTitle('Ghost Admin', 'Ghost admin has no title');
@ -338,6 +338,17 @@ CasperTest.begin('Publish menu - existing post', 19, function suite(test) {
test.assertSelectorHasText('.entry-preview .rendered-markdown', 'test', 'Editor value is correct'); test.assertSelectorHasText('.entry-preview .rendered-markdown', 'test', 'Editor value is correct');
}); });
casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
casper.waitForOpaque('.js-publish-splitbutton .dropdown-menu', function onSuccess() {
test.assert(true, 'popup menu should be visible after clicking post-settings icon');
test.assertNotVisible(
'.js-publish-splitbutton .delete', 'delete post button shouldn\'t be visible on unsaved drafts'
);
});
casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
// Create a post in draft status // Create a post in draft status
casper.thenClick('.js-publish-button'); casper.thenClick('.js-publish-button');
@ -353,6 +364,17 @@ CasperTest.begin('Publish menu - existing post', 19, function suite(test) {
test.assertSelectorHasText('.js-publish-button', 'Save Draft'); test.assertSelectorHasText('.js-publish-button', 'Save Draft');
}); });
casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
casper.waitForOpaque('.js-publish-splitbutton .open', function onSuccess() {
test.assert(true, 'delete post button should be visible for saved drafts');
test.assertVisible(
'.js-publish-splitbutton .delete', 'delete post button should be visible on saved drafts'
);
});
casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
casper.then(function switchMenuToPublish() { casper.then(function switchMenuToPublish() {
// Open the publish options menu; // Open the publish options menu;
casper.thenClick('.js-publish-splitbutton .dropdown-toggle'); casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
@ -417,6 +439,58 @@ CasperTest.begin('Publish menu - existing post', 19, function suite(test) {
}); });
}); });
CasperTest.begin('Publish menu - delete post', 7, function testDeleteModal(test) {
// Create a post that can be deleted
CasperTest.Routines.createTestPost.run(false);
// Begin test
casper.thenOpenAndWaitForPageLoad('content', function testTitleAndUrl() {
test.assertTitle('Ghost Admin', 'Title is "Ghost Admin"');
test.assertUrlMatch(/ghost\/\d+\/$/, 'Landed on the correct URL');
});
// Transition to the editor
casper.thenClick('.post-edit');
casper.waitForSelector('#entry-title');
// Open post settings menu
casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
casper.waitForOpaque('.js-publish-splitbutton .open');
casper.thenClick('.js-publish-splitbutton li:nth-child(4) a');
casper.waitUntilVisible('#modal-container', function onSuccess() {
test.assertSelectorHasText(
'.modal-content .modal-header',
'Are you sure you want to delete this post?',
'delete modal has correct text');
});
casper.thenClick('.js-button-reject');
casper.waitWhileVisible('#modal-container', function onSuccess() {
test.assert(true, 'clicking cancel should close the delete post modal');
});
// Test delete
casper.thenClick('.js-publish-splitbutton .dropdown-toggle');
casper.waitForOpaque('.js-publish-splitbutton .open');
casper.thenClick('.js-publish-splitbutton li:nth-child(4) a');
casper.waitForSelector('#modal-container .modal-content', function onSuccess() {
test.assertExists('.modal-content .js-button-accept', 'delete button exists');
// Delete the post
this.click('.modal-content .js-button-accept');
casper.waitForSelector('.notification-success', function onSuccess() {
test.assert(true, 'Got success notification from delete post');
test.assertSelectorHasText('.notification-message', 'Your post has been deleted.');
}, function onTimeout() {
test.fail('No success notification from delete post');
});
});
});
CasperTest.begin('Publish menu - new post status is correct after failed save', 4, function suite(test) { CasperTest.begin('Publish menu - new post status is correct after failed save', 4, function suite(test) {
casper.thenOpenAndWaitForPageLoad('editor', function testTitleAndUrl() { casper.thenOpenAndWaitForPageLoad('editor', function testTitleAndUrl() {
test.assertTitle('Ghost Admin', 'Ghost admin has no title'); test.assertTitle('Ghost Admin', 'Ghost admin has no title');

View file

@ -3,7 +3,7 @@
/*globals CasperTest, casper, __utils__ */ /*globals CasperTest, casper, __utils__ */
CasperTest.begin('Post settings menu', 14, function suite(test) { CasperTest.begin('Post settings menu', 10, function suite(test) {
casper.thenOpenAndWaitForPageLoad('editor', function testTitleAndUrl() { casper.thenOpenAndWaitForPageLoad('editor', function testTitleAndUrl() {
test.assertTitle('Ghost Admin', 'Ghost admin has no title'); test.assertTitle('Ghost Admin', 'Ghost admin has no title');
test.assertUrlMatch(/ghost\/editor\/$/, 'Landed on the correct URL'); test.assertUrlMatch(/ghost\/editor\/$/, 'Landed on the correct URL');
@ -15,20 +15,8 @@ CasperTest.begin('Post settings menu', 14, function suite(test) {
test.assertExists('.post-settings-menu #url', 'url field exists'); test.assertExists('.post-settings-menu #url', 'url field exists');
test.assertExists('.post-settings-menu .post-setting-date', 'publication date field exists'); test.assertExists('.post-settings-menu .post-setting-date', 'publication date field exists');
test.assertExists('.post-settings-menu .post-setting-static-page', 'static page checkbox field exists'); test.assertExists('.post-settings-menu .post-setting-static-page', 'static page checkbox field exists');
test.assertExists('.post-settings-menu button.delete', 'delete post button exists');
}); });
casper.thenClick('.post-settings');
casper.waitForOpaque('.post-settings-menu', function onSuccess() {
test.assert(true, 'popup menu should be visible after clicking post-settings icon');
test.assertNotVisible(
'.post-settings-menu button.delete', 'delete post button shouldn\'t be visible on unsaved drafts'
);
});
casper.thenClick('.post-settings');
// Enter a title and save draft so converting to/from static post // Enter a title and save draft so converting to/from static post
// will result in notifications and 'Delete This Post' button appears // will result in notifications and 'Delete This Post' button appears
casper.then(function () { casper.then(function () {
@ -51,60 +39,6 @@ CasperTest.begin('Post settings menu', 14, function suite(test) {
casper.waitForOpaque('.post-settings-menu', function onSuccess() { casper.waitForOpaque('.post-settings-menu', function onSuccess() {
test.assert(true, 'post settings menu should be visible after clicking post-settings icon'); test.assert(true, 'post settings menu should be visible after clicking post-settings icon');
}); });
casper.waitUntilVisible('.post-settings-menu button.delete', function onSuccess() {
test.assert(true, 'delete post button should be visible for saved drafts');
});
});
CasperTest.begin('Delete post modal', 7, function testDeleteModal(test) {
// Create a post that can be deleted
CasperTest.Routines.createTestPost.run(false);
// Begin test
casper.thenOpenAndWaitForPageLoad('content', function testTitleAndUrl() {
test.assertTitle('Ghost Admin', 'Title is "Ghost Admin"');
test.assertUrlMatch(/ghost\/\d+\/$/, 'Landed on the correct URL');
});
// Transition to the editor
casper.thenClick('.post-edit');
casper.waitForSelector('#entry-title');
// Open post settings menu
casper.thenClick('.post-settings');
casper.thenClick('.post-settings-menu button.delete');
casper.waitUntilVisible('#modal-container', function onSuccess() {
test.assertSelectorHasText(
'.modal-content .modal-header',
'Are you sure you want to delete this post?',
'delete modal has correct text');
});
casper.thenClick('.js-button-reject');
casper.waitWhileVisible('#modal-container', function onSuccess() {
test.assert(true, 'clicking cancel should close the delete post modal');
});
// Test delete
casper.thenClick('.post-settings');
casper.thenClick('.post-settings-menu button.delete');
casper.waitForSelector('#modal-container .modal-content', function onSuccess() {
test.assertExists('.modal-content .js-button-accept', 'delete button exists');
// Delete the post
this.click('.modal-content .js-button-accept');
casper.waitForSelector('.notification-success', function onSuccess() {
test.assert(true, 'Got success notification from delete post');
test.assertSelectorHasText('.notification-message', 'Your post has been deleted.');
}, function onTimeout() {
test.fail('No success notification from delete post');
});
});
}); });
CasperTest.begin('Post url can be changed', 4, function suite(test) { CasperTest.begin('Post url can be changed', 4, function suite(test) {