From 785ca2b2de0f896a3419a0c9a11b5117807b6629 Mon Sep 17 00:00:00 2001 From: webchick <webchick@24967.no-reply.drupal.org> Date: Tue, 29 Jan 2013 19:30:40 -0800 Subject: [PATCH] Issue #1751606 follow-up by sun, swentel: Various fixes for content creation page. --- core/includes/form.inc | 46 +++++ core/includes/theme.inc | 29 +-- core/misc/dropbutton/dropbutton.base.css | 12 +- .../lib/Drupal/node/NodeFormController.php | 141 ++++++--------- .../Drupal/node/Tests/NodeFormButtonsTest.php | 165 ++++++++++++++++++ core/modules/node/node.admin.css | 127 -------------- core/modules/node/node.admin.inc | 3 - .../system/Tests/Theme/FunctionsTest.php | 18 +- core/modules/system/system.module | 2 +- core/themes/seven/style.css | 111 ++++++++++++ 10 files changed, 410 insertions(+), 244 deletions(-) create mode 100644 core/modules/node/lib/Drupal/node/Tests/NodeFormButtonsTest.php diff --git a/core/includes/form.inc b/core/includes/form.inc index 513615cdd0a9..3684578f9f42 100644 --- a/core/includes/form.inc +++ b/core/includes/form.inc @@ -3336,6 +3336,52 @@ function form_process_actions($element, &$form_state) { return $element; } +/** + * #pre_render callback for #type 'actions'. + * + * This callback iterates over all child elements of the #type 'actions' + * container to look for elements with a #dropbutton property, so as to group + * those elements into dropbuttons. As such, it works similar to #group, but is + * specialized for dropbuttons. + * + * The value of #dropbutton denotes the dropbutton to group the child element + * into. For example, two different values of 'foo' and 'bar' on child elements + * would generate two separate dropbuttons, which each contain the corresponding + * buttons. + * + * @param array $element + * The #type 'actions' element to process. + * + * @return array + * The processed #type 'actions' element, including individual buttons grouped + * into new #type 'dropbutton' elements. + */ +function form_pre_render_actions_dropbutton(array $element) { + $dropbuttons = array(); + foreach (element_children($element, TRUE) as $key) { + if (isset($element[$key]['#dropbutton'])) { + $dropbutton = $element[$key]['#dropbutton']; + // If there is no dropbutton for this button group yet, create one. + if (!isset($dropbuttons[$dropbutton])) { + $dropbuttons[$dropbutton] = array( + '#type' => 'dropbutton', + ); + } + // Add this button to the corresponding dropbutton. + // @todo Change #type 'dropbutton' to be based on theme_item_list() + // instead of theme_links() to avoid this preemptive rendering. + $button = drupal_render($element[$key]); + $dropbuttons[$dropbutton]['#links'][$key] = array( + 'title' => $button, + 'html' => TRUE, + ); + } + } + // @todo For now, all dropbuttons appear first. Consider to invent a more + // fancy sorting/injection algorithm here. + return $dropbuttons + $element; +} + /** * #process callback for #pattern form element property. * diff --git a/core/includes/theme.inc b/core/includes/theme.inc index 79a0b87e4628..3483330a772c 100644 --- a/core/includes/theme.inc +++ b/core/includes/theme.inc @@ -1802,11 +1802,11 @@ function theme_links($variables) { // Merge in default array properties into $link. $link += array( 'html' => FALSE, - 'attributes' => array(), ); - $item = '<span' . new Attribute($link['attributes']) . '>'; - $item .= ($link['html'] ? $link['title'] : check_plain($link['title'])); - $item .= '</span>'; + $item = ($link['html'] ? $link['title'] : check_plain($link['title'])); + if (isset($link['attributes'])) { + $item = '<span' . new Attribute($link['attributes']) . '>' . $item . '</span>'; + } } $output .= '<li' . new Attribute(array('class' => $class)) . '>'; @@ -1834,24 +1834,6 @@ function theme_dropbutton_wrapper($variables) { } } -/** - * Returns HTML for wrapping a dropbutton list. - * - * Use this function if the dropbutton contains submit buttons. These elements - * need to have a #prefix and #suffix element that wraps those into an <li> - * element. - * - * @param array $variables - * An associative array containing: - * - element: An associative array containing the properties and children of - * the dropbutton list. Properties used: #children. - */ -function theme_dropbutton_list_wrapper($variables) { - if (!empty($variables['element']['#children'])) { - return '<ul class="dropbutton">' . $variables['element']['#children'] . '</ul>'; - } -} - /** * Returns HTML for an image. * @@ -3163,9 +3145,6 @@ function drupal_common_theme() { 'dropbutton_wrapper' => array( 'render element' => 'element', ), - 'dropbutton_list_wrapper' => array( - 'render element' => 'element', - ), 'image' => array( // HTML 4 and XHTML 1.0 always require an alt attribute. The HTML 5 draft // allows the alt attribute to be omitted in some cases. Therefore, diff --git a/core/misc/dropbutton/dropbutton.base.css b/core/misc/dropbutton/dropbutton.base.css index 9c1a3ef7716f..35e9692dd220 100644 --- a/core/misc/dropbutton/dropbutton.base.css +++ b/core/misc/dropbutton/dropbutton.base.css @@ -22,13 +22,21 @@ .js .dropbutton-widget { max-width: 100%; } - @media screen and (max-width:600px) { .js .dropbutton-wrapper { width: 100%; } } +/* Splitbuttons */ +.form-actions .dropbutton-wrapper { + float: left; +} +.js .form-actions .dropbutton-widget { + position: static; +} + + .js .dropbutton-widget { position: absolute; } @@ -79,7 +87,7 @@ text-indent: 110%; top: 0; white-space: nowrap; - width: 2.08em; + width: 2em; } .dropbutton-toggle button { background: none; diff --git a/core/modules/node/lib/Drupal/node/NodeFormController.php b/core/modules/node/lib/Drupal/node/NodeFormController.php index a7354eecb3f7..d1b32dc10c9d 100644 --- a/core/modules/node/lib/Drupal/node/NodeFormController.php +++ b/core/modules/node/lib/Drupal/node/NodeFormController.php @@ -275,99 +275,70 @@ public function form(array $form, array &$form_state, EntityInterface $node) { } /** - * Overrides Drupal\entity\EntityFormController::actionsElement(). + * Overrides Drupal\Core\Entity\EntityFormController::actions(). */ - protected function actionsElement(array $form, array &$form_state) { - $element = parent::actionsElement($form, $form_state); + protected function actions(array $form, array &$form_state) { + $element = parent::actions($form, $form_state); $node = $this->getEntity($form_state); + $preview_mode = variable_get('node_preview_' . $node->type, DRUPAL_OPTIONAL); - // Because some of the 'links' are actually submit buttons, we have to - // manually wrap each item in <li> and the whole list in <ul>. The - // <ul> is added with a #theme_wrappers function. - $element['operations'] = array( - '#type' => 'operations', - '#subtype' => 'node', - '#attached' => array ( - 'css' => array( - drupal_get_path('module', 'node') . '/node.admin.css', - ), - ), - ); - - $element['operations']['actions'] = array( - '#theme_wrappers' => array('dropbutton_list_wrapper') - ); - - // Depending on the state of the node (published or unpublished) and - // whether the current user has the permission to change the status, the - // labels and order of the buttons will vary. - if (user_access('administer nodes')) { - $element['operations']['actions']['publish'] = array( - '#type' => 'submit', - '#value' => t('Save and publish'), - '#submit' => array(array($this, 'publish'), array($this, 'submit'), array($this, 'save')), - '#validate' => array(array($this, 'validate')), - '#button_type' => $node->status ? 'primary' : '', - '#weight' => 0, - '#prefix' => '<li class="publish">', - '#suffix' => '</li>', - ); - $element['operations']['actions']['unpublish'] = array( - '#type' => 'submit', - '#value' => t('Save as unpublished'), - '#submit' => array(array($this, 'unpublish'), array($this, 'submit'), array($this, 'save')), - '#validate' => array(array($this, 'validate')), - '#button_type' => empty($node->status) ? 'primary' : '', - '#weight' => $node->status ? 1 : -1, - '#prefix' => '<li class="unpublish">', - "#suffix" => '</li>', - ); + $element['submit']['#access'] = $preview_mode != DRUPAL_REQUIRED || (!form_get_errors() && isset($form_state['node_preview'])); - if (!empty($node->nid)) { - if ($node->status) { - $publish_label = t('Save and keep published'); - $unpublish_label = t('Save and unpublish'); - } - else { - $publish_label = t('Save and publish'); - $unpublish_label = t('Save and keep unpublished'); - } - $element['operations']['actions']['publish']['#value'] = $publish_label; - $element['operations']['actions']['unpublish']['#value'] = $unpublish_label; + // If saving is an option, privileged users get dedicated form submit + // buttons to adjust the publishing status while saving in one go. + // @todo This adjustment makes it close to impossible for contributed + // modules to integrate with "the Save operation" of this form. Modules + // need a way to plug themselves into 1) the ::submit() step, and + // 2) the ::save() step, both decoupled from the pressed form button. + if ($element['submit']['#access'] && user_access('administer nodes')) { + // isNew | prev status » default & publish label & unpublish label + // 1 | 1 » publish & Save and publish & Save as unpublished + // 1 | 0 » unpublish & Save and publish & Save as unpublished + // 0 | 1 » publish & Save and keep published & Save and unpublish + // 0 | 0 » unpublish & Save and keep unpublished & Save and publish + + // Add a "Publish" button. + $element['publish'] = $element['submit']; + $element['publish']['#dropbutton'] = 'save'; + if ($node->isNew()) { + $element['publish']['#value'] = t('Save and publish'); } - } - // The user has no permission to change the status of the node. Just - // show a save button without the 'publish' or 'unpublish' callback in - // the #submit definition. - else { - $element['operations']['actions']['save'] = array( - '#type' => 'submit', - '#value' => t('Save'), - '#submit' => array(array($this, 'submit'), array($this, 'save')), - '#validate' => array(array($this, 'validate')), - '#button_type' => 'primary', - '#weight' => 1, - '#prefix' => '<li class="save">', - "#suffix" => '</li>', - ); - } - - unset($element['submit']); + else { + $element['publish']['#value'] = $node->status ? t('Save and keep published') : t('Save and publish'); + } + $element['publish']['#weight'] = 0; + array_unshift($element['publish']['#submit'], array($this, 'publish')); + + // Add a "Unpublish" button. + $element['unpublish'] = $element['submit']; + $element['unpublish']['#dropbutton'] = 'save'; + if ($node->isNew()) { + $element['unpublish']['#value'] = t('Save as unpublished'); + } + else { + $element['unpublish']['#value'] = !$node->status ? t('Save and keep unpublished') : t('Save and unpublish'); + } + $element['unpublish']['#weight'] = 10; + array_unshift($element['unpublish']['#submit'], array($this, 'unpublish')); - return $element; - } + // If already published, the 'publish' button is primary. + if ($node->status) { + unset($element['unpublish']['#button_type']); + } + // Otherwise, the 'unpublish' button is primary and should come first. + else { + unset($element['publish']['#button_type']); + $element['unpublish']['#weight'] = -10; + } - /* - * Overrides Drupal\Core\Entity\EntityFormController::actions(). - */ - protected function actions(array $form, array &$form_state) { - $element = parent::actions($form, $form_state); - $node = $this->getEntity($form_state); - $preview_mode = variable_get('node_preview_' . $node->type, DRUPAL_OPTIONAL); + // Remove the "Save" button. + $element['submit']['#access'] = FALSE; + } $element['preview'] = array( '#access' => $preview_mode != DRUPAL_DISABLED, '#value' => t('Preview'), + '#weight' => 20, '#validate' => array( array($this, 'validate'), ), @@ -377,8 +348,8 @@ protected function actions(array $form, array &$form_state) { ), ); - $element['submit']['#access'] = $preview_mode != DRUPAL_REQUIRED || (!form_get_errors() && isset($form_state['node_preview'])); $element['delete']['#access'] = node_access('delete', $node); + $element['delete']['#weight'] = 100; return $element; } @@ -473,7 +444,7 @@ public function preview(array $form, array &$form_state) { */ public function publish(array $form, array &$form_state) { $node = $this->getEntity($form_state); - $node->status = TRUE; + $node->status = 1; return $node; } @@ -487,7 +458,7 @@ public function publish(array $form, array &$form_state) { */ public function unpublish(array $form, array &$form_state) { $node = $this->getEntity($form_state); - $node->status = FALSE; + $node->status = 0; return $node; } diff --git a/core/modules/node/lib/Drupal/node/Tests/NodeFormButtonsTest.php b/core/modules/node/lib/Drupal/node/Tests/NodeFormButtonsTest.php new file mode 100644 index 000000000000..a2bd56c8b92d --- /dev/null +++ b/core/modules/node/lib/Drupal/node/Tests/NodeFormButtonsTest.php @@ -0,0 +1,165 @@ +<?php + +/** + * @file + * Contains Drupal\node\Tests\NodeLoadHooksTest. + */ + +namespace Drupal\node\Tests; + +/** + * Tests the node form buttons. + */ +class NodeFormButtonsTest extends NodeTestBase { + + protected $web_user; + + protected $admin_user; + + public static function getInfo() { + return array( + 'name' => 'Node form buttons', + 'description' => 'Test all the different buttons on the node form.', + 'group' => 'Node', + ); + } + + function setUp() { + parent::setUp(); + + // Create a user that has no access to change the state of the node. + $this->web_user = $this->drupalCreateUser(array('create article content', 'edit own article content')); + // Create a user that has access to change the state of the node. + $this->admin_user = $this->drupalCreateUser(array('administer nodes', 'bypass node access')); + } + + /** + * Tests that the right buttons are displayed for saving nodes. + */ + function testNodeFormButtons() { + + // Login as administrative user. + $this->drupalLogin($this->admin_user); + + // Verify the buttons on a node add form. + $this->drupalGet('node/add/article'); + $this->assertButtons(array(t('Save and publish'), t('Save as unpublished'))); + + // Save the node and assert it's published after clicking + // 'Save and publish'. + $edit = array('title' => $this->randomString()); + $this->drupalPost('node/add/article', $edit, t('Save and publish')); + + // Get the node. + $node_1 = node_load(1); + $this->assertEqual(1, $node_1->status, 'Node is published'); + + // Verify the buttons on a node edit form. + $this->drupalGet('node/' . $node_1->nid . '/edit'); + $this->assertButtons(array(t('Save and keep published'), t('Save and unpublish'))); + + // Save the node and verify it's still published after clicking + // 'Save and keep published'. + $this->drupalPost(NULL, $edit, t('Save and keep published')); + $node = node_load(1, TRUE); + $this->assertEqual(1, $node_1->status, 'Node is published'); + + // Save the node and verify it's unpublished after clicking + // 'Save and unpublish'. + $this->drupalPost('node/' . $node_1->nid . '/edit', $edit, t('Save and unpublish')); + $node_1 = node_load(1, TRUE); + $this->assertEqual(0, $node_1->status, 'Node is unpublished'); + + // Verify the buttons on an unpublished node edit screen. + $this->drupalGet('node/' . $node_1->nid . '/edit'); + $this->assertButtons(array(t('Save and keep unpublished'), t('Save and publish'))); + + // Create a node as a normal user. + $this->drupalLogout(); + $this->drupalLogin($this->web_user); + + // Verify the buttons for a normal user. + $this->drupalGet('node/add/article'); + $this->assertButtons(array(t('Save')), FALSE); + + // Create the node. + $edit = array('title' => $this->randomString()); + $this->drupalPost('node/add/article', $edit, t('Save')); + $node_2 = node_load(2); + $this->assertEqual(1, $node_2->status, 'Node is published'); + + // Login as an administrator and unpublish the node that just + // was created by the normal user. + $this->drupalLogout(); + $this->drupalLogin($this->admin_user); + $this->drupalPost('node/' . $node_2->nid . '/edit', array(), t('Save and unpublish')); + $node_2 = node_load(2, TRUE); + $this->assertEqual(0, $node_2->status, 'Node is unpublished'); + + // Login again as the normal user, save the node and verify + // it's still unpublished. + $this->drupalLogout(); + $this->drupalLogin($this->web_user); + $this->drupalPost('node/' . $node_2->nid . '/edit', array(), t('Save')); + $node_2 = node_load(2, TRUE); + $this->assertEqual(0, $node_2->status, 'Node is still unpublished'); + $this->drupalLogout(); + + // Set article content type default to unpublished. This will change the + // the initial order of buttons and/or status of the node when creating + // a node. + variable_set('node_options_article', array('promote')); + $this->refreshVariables(); + + // Verify the buttons on a node add form for an administrator. + $this->drupalLogin($this->admin_user); + $this->drupalGet('node/add/article'); + $this->assertButtons(array(t('Save as unpublished'), t('Save and publish'))); + + // Verify the node is unpublished by default for a normal user. + $this->drupalLogout(); + $this->drupalLogin($this->web_user); + $edit = array('title' => $this->randomString()); + $this->drupalPost('node/add/article', $edit, t('Save')); + $node_3 = node_load(3); + $this->assertEqual(0, $node_3->status, 'Node is unpublished'); + } + + /** + * Assert method to verify the buttons in the dropdown element. + * + * @param array $buttons + * A collection of buttons to assert for on the page. + * @param bool $dropbutton + * Whether to check if the buttons are in a dropbutton widget or not. + */ + public function assertButtons($buttons, $dropbutton = TRUE) { + + // Try to find a Save button. + $save_button = $this->xpath('//input[@type="submit"][@value="Save"]'); + + // Verify that the number of buttons passed as parameters is + // available in the dropbutton widget. + if ($dropbutton) { + $i = 0; + $count = count($buttons); + + // Assert there is no save button. + $this->assertTrue(empty($save_button)); + + // Dropbutton elements. + $elements = $this->xpath('//div[@class="dropbutton-wrapper"]//input[@type="submit"]'); + $this->assertEqual($count, count($elements)); + foreach ($elements as $element) { + $value = isset($element['value']) ? (string) $element['value'] : ''; + $this->assertEqual($buttons[$i], $value); + $i++; + } + } + else { + // Assert there is a save button. + $this->assertTrue(!empty($save_button)); + $this->assertNoRaw('dropbutton-wrapper'); + } + } +} diff --git a/core/modules/node/node.admin.css b/core/modules/node/node.admin.css index 9ef08e1e4305..101a38d53904 100644 --- a/core/modules/node/node.admin.css +++ b/core/modules/node/node.admin.css @@ -9,130 +9,3 @@ .revision-current { background: #ffc; } - -/** - * Node form dropbuttons. - */ -.form-actions .dropbutton-wrapper { - float: left; - margin-right: 1em; -} - -.form-actions .dropbutton-wrapper .dropbutton-widget { - position: static; -} - -.form-actions .dropbutton-wrapper li a, -.form-actions .dropbutton-wrapper input { - padding: 5px 17px 6px 17px; - margin-bottom: 0em; - border: medium; - border-radius: 0; - background: none; -} - -.form-actions .dropbutton-wrapper input:hover { - background: none; - border: none; -} - -.form-actions .button { - background: #fefefe; - background-image: -webkit-linear-gradient(top, #fefefe, #e0e0e0); - background-image: -moz-linear-gradient(top, #fefefe, #e0e0e0); - background-image: -o-linear-gradient(top, #fefefe, #e0e0e0); - background-image: linear-gradient(to bottom, #fefefe, #e0e0e0); - border: 1px solid #c8c8c8; - border-radius: 3px; - text-decoration: none; - padding: 6px 17px 6px 17px; - margin-left: 0; -} - -.form-actions .button:focus, -.form-actions .button:hover { - background: #fefefe; - background-image: -webkit-linear-gradient(top, #fefefe, #eaeaea); - background-image: -moz-linear-gradient(top, #fefefe, #eaeaea); - background-image: -o-linear-gradient(top, #fefefe, #eaeaea); - background-image: linear-gradient(to bottom, #fefefe, #eaeaea); - -webkit-box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); - box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); - color: #2e2e2e; - text-decoration: none; -} -.form-actions .button:active { - border: 1px solid #c8c8c8; - background: #fefefe; - background-image: -webkit-linear-gradient(top, #eaeaea, #fefefe); - background-image: -moz-linear-gradient(top, #eaeaea, #fefefe); - background-image: -o-linear-gradient(top, #eaeaea, #fefefe); - background-image: linear-gradient(to bottom, #eaeaea, #fefefe); - -webkit-box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); - box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); - color: #2e2e2e; - text-decoration: none; - text-shadow: none; -} - -/* Delete button */ -.form-actions .button-danger { - color: #c72100; - background: none; - border: none; - float: right; - margin-right: 0; - margin-left: 0; - padding-right: 0; - padding-left: 0; -} -.form-actions .button-danger:hover, -.form-actions .button-danger:focus { - color: #ff2a00; - background: none; - border: none; - text-decoration: underline; -} -.form-actions .button-danger:active { - color: #ff2a00; - background: none; - border: none; - text-decoration: underline; -} - - -/** - * Form edit action theming - */ -.js .form-actions .dropbutton-widget { - background-color: #50a0e9; - background-image: -moz-linear-gradient(-90deg, #50a0e9, #4481dc); - background-image: -o-linear-gradient(-90deg, #50a0e9, #4481dc); - background-image: -webkit-linear-gradient(-90deg, #50a0e9, #4481dc); - background-image: linear-gradient(180deg, #50a0e9, #4481dc); - border-radius: 3px; - border: 1px solid #3974ae; -} -.js .form-actions .dropbutton-widget .dropbutton li { - border-top: 1px solid rgba(255, 255, 255, 0.5); - border-top-left-radius: 3px; -} -.js .form-actions .dropbutton-widget .dropbutton .dropbutton-toggle { - border-top-left-radius: 0px; - border-top-right-radius: 3px; - top: 1px; -} -.js .form-actions .dropbutton-widget .dropbutton .secondary-action { - border-top: 1px solid rgba(255, 255, 255, 0.3); - border-top-left-radius: 0px; -} -.js .form-actions .dropbutton-widget .button { - color: #ffffff; - text-shadow: 1px 1px 1px rgba(31, 83, 131, 0.8); -} -.js .form-actions .dropbutton-multiple.open .dropbutton-action:hover { - background-color: #50a0e9; -} - - - diff --git a/core/modules/node/node.admin.inc b/core/modules/node/node.admin.inc index 10a12c6eec9a..7c360a4d3d4f 100644 --- a/core/modules/node/node.admin.inc +++ b/core/modules/node/node.admin.inc @@ -434,9 +434,6 @@ function node_admin_nodes() { '#title' => t('Update options'), '#attributes' => array('class' => array('container-inline')), '#access' => $admin_access, - '#attached' => array ( - 'css' => array(drupal_get_path('module', 'node') . '/css/node-admin.theme.css'), - ), ); $options = array(); foreach (module_invoke_all('node_operations') as $operation => $array) { diff --git a/core/modules/system/lib/Drupal/system/Tests/Theme/FunctionsTest.php b/core/modules/system/lib/Drupal/system/Tests/Theme/FunctionsTest.php index d3a090f71a77..1faf6d86fdf9 100644 --- a/core/modules/system/lib/Drupal/system/Tests/Theme/FunctionsTest.php +++ b/core/modules/system/lib/Drupal/system/Tests/Theme/FunctionsTest.php @@ -164,7 +164,7 @@ function testLinks() { $expected_links = ''; $expected_links .= '<ul id="somelinks">'; $expected_links .= '<li class="a-link odd first"><a href="' . url('a/link') . '">' . check_plain('A <link>') . '</a></li>'; - $expected_links .= '<li class="plain-text even"><span>' . check_plain('Plain "text"') . '</span></li>'; + $expected_links .= '<li class="plain-text even">' . check_plain('Plain "text"') . '</li>'; $expected_links .= '<li class="front-page odd last active"><a href="' . url('<front>') . '" class="active">' . check_plain('Front page') . '</a></li>'; $expected_links .= '</ul>'; @@ -185,6 +185,22 @@ function testLinks() { $expected_heading = '<h3 id="heading">Links heading</h3>'; $expected = $expected_heading . $expected_links; $this->assertThemeOutput('links', $variables, $expected); + + // Verify that passing attributes for the links work. + $variables['links']['a link']['attributes'] = array( + 'class' => array('a/class'), + ); + $variables['links']['plain text']['attributes'] = array( + 'class' => array('a/class'), + ); + $expected_links = ''; + $expected_links .= '<ul id="somelinks">'; + $expected_links .= '<li class="a-link odd first"><a href="' . url('a/link') . '" class="a/class">' . check_plain('A <link>') . '</a></li>'; + $expected_links .= '<li class="plain-text even"><span class="a/class">' . check_plain('Plain "text"') . '</span></li>'; + $expected_links .= '<li class="front-page odd last active"><a href="' . url('<front>') . '" class="active">' . check_plain('Front page') . '</a></li>'; + $expected_links .= '</ul>'; + $expected = $expected_heading . $expected_links; + $this->assertThemeOutput('links', $variables, $expected); } /** diff --git a/core/modules/system/system.module b/core/modules/system/system.module index 106ef14f176b..c621915432f4 100644 --- a/core/modules/system/system.module +++ b/core/modules/system/system.module @@ -576,7 +576,7 @@ function system_element_info() { '#theme_wrappers' => array('container'), ); $types['actions'] = array( - '#process' => array('form_process_actions', 'form_process_container'), + '#process' => array('form_pre_render_actions_dropbutton', 'form_process_actions', 'form_process_container'), '#weight' => 100, '#theme_wrappers' => array('container'), ); diff --git a/core/themes/seven/style.css b/core/themes/seven/style.css index 83d52931d469..a71a8ffd8541 100644 --- a/core/themes/seven/style.css +++ b/core/themes/seven/style.css @@ -1537,3 +1537,114 @@ details.fieldset-no-legend { .entity-meta details .summary { display: none; /* Hide JS summaries. @todo Rethink summaries. */ } + + +/** + * Node form dropbuttons. + */ +.form-actions .dropbutton-wrapper li a, +.form-actions .dropbutton-wrapper input { + padding: 5px 17px 6px 17px; + margin-bottom: 0em; + border: medium; + border-radius: 0; + background: none; +} +.form-actions .dropbutton-wrapper input:hover { + background: none; + border: none; +} +.form-actions .button { + background: #fefefe; + background-image: -webkit-linear-gradient(top, #fefefe, #e0e0e0); + background-image: -moz-linear-gradient(top, #fefefe, #e0e0e0); + background-image: -o-linear-gradient(top, #fefefe, #e0e0e0); + background-image: linear-gradient(to bottom, #fefefe, #e0e0e0); + border: 1px solid #c8c8c8; + border-radius: 3px; + text-decoration: none; + padding: 6px 17px 6px 17px; + margin-left: 0; +} +.form-actions .button:focus, +.form-actions .button:hover { + background: #fefefe; + background-image: -webkit-linear-gradient(top, #fefefe, #eaeaea); + background-image: -moz-linear-gradient(top, #fefefe, #eaeaea); + background-image: -o-linear-gradient(top, #fefefe, #eaeaea); + background-image: linear-gradient(to bottom, #fefefe, #eaeaea); + -webkit-box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); + box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); + color: #2e2e2e; + text-decoration: none; +} +.form-actions .button:active { + border: 1px solid #c8c8c8; + background: #fefefe; + background-image: -webkit-linear-gradient(top, #eaeaea, #fefefe); + background-image: -moz-linear-gradient(top, #eaeaea, #fefefe); + background-image: -o-linear-gradient(top, #eaeaea, #fefefe); + background-image: linear-gradient(to bottom, #eaeaea, #fefefe); + -webkit-box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); + box-shadow: 1px 1px 3px rgba(50, 50, 50, 0.1); + color: #2e2e2e; + text-decoration: none; + text-shadow: none; +} +/* Delete button */ +.form-actions .button-danger { + color: #c72100; + background: none; + border: none; + float: right; + margin-right: 0; + margin-left: 0; + padding-right: 0; + padding-left: 0; +} +.form-actions .button-danger:hover, +.form-actions .button-danger:focus { + color: #ff2a00; + background: none; + border: none; + text-decoration: underline; +} +.form-actions .button-danger:active { + color: #ff2a00; + background: none; + border: none; + text-decoration: underline; +} + +/** + * Form edit action theming + */ +.js .form-actions .dropbutton-widget { + background-color: #50a0e9; + background-image: -moz-linear-gradient(-90deg, #50a0e9, #4481dc); + background-image: -o-linear-gradient(-90deg, #50a0e9, #4481dc); + background-image: -webkit-linear-gradient(-90deg, #50a0e9, #4481dc); + background-image: linear-gradient(180deg, #50a0e9, #4481dc); + border-radius: 3px; + border: 1px solid #3974ae; +} +.js .form-actions .dropbutton-widget .dropbutton li { + border-top: 1px solid rgba(255, 255, 255, 0.5); + border-top-left-radius: 3px; +} +.js .form-actions .dropbutton-widget .dropbutton .dropbutton-toggle { + border-top-left-radius: 0px; + border-top-right-radius: 3px; + top: 1px; +} +.js .form-actions .dropbutton-widget .dropbutton .secondary-action { + border-top: 1px solid rgba(255, 255, 255, 0.3); + border-top-left-radius: 0px; +} +.js .form-actions .dropbutton-widget .button { + color: #ffffff; + text-shadow: 1px 1px 1px rgba(31, 83, 131, 0.8); +} +.js .form-actions .dropbutton-multiple.open .dropbutton-action:hover { + background-color: #50a0e9; +} -- GitLab