From 5a49056a32c5226149cee8dd0ef9b73516c4e7f6 Mon Sep 17 00:00:00 2001 From: Earl Miles <merlin@logrus.com> Date: Thu, 27 Nov 2008 17:29:35 +0000 Subject: [PATCH] Add callbacks to the collapsible so that it can be trivially transformed into other things like an accordion. Add accordion functionality to the task admin UI. --- css/collapsible-div.css | 4 +++ delegator/css/task-handlers.css | 17 ++++++++++++ delegator/delegator.admin.inc | 11 +++++++- delegator/delegator.module | 9 +++++++ delegator/help/api-task-handler.html | 6 ++++- delegator/js/task-handlers.js | 39 +++++++++++++++++++++++++++- help/plugins-creating.html | 13 +++++++++- includes/plugins.inc | 25 +++++++++++++++--- js/collapsible-div.js | 32 +++++++++++++++++------ 9 files changed, 141 insertions(+), 15 deletions(-) diff --git a/css/collapsible-div.css b/css/collapsible-div.css index 0f900474..e241128d 100644 --- a/css/collapsible-div.css +++ b/css/collapsible-div.css @@ -10,6 +10,10 @@ background-image: url(../images/collapsible-expanded.png); } +.ctools-collapsible-container .ctools-collapsible-handle { + cursor: pointer; +} + .ctools-collapsible-container .ctools-toggle-collapsed { background-image: url(../images/collapsible-collapsed.png); } diff --git a/delegator/css/task-handlers.css b/delegator/css/task-handlers.css index c1624c07..b937812c 100644 --- a/delegator/css/task-handlers.css +++ b/delegator/css/task-handlers.css @@ -46,3 +46,20 @@ padding-right: .75em; } +#delegator-task-list-arrange td { + vertical-align: top; +} + +#delegator-task-list-arrange tr.delegator-collapsible td { + padding-left: 4em; +} + +#delegator-task-list-arrange tr.draggable .ctools-collapsible-content { + display: none; +} + +/* +#delegator-task-list-arrange tr.delegator-collapsible div.ctools-collapsible-content { + display: block; +} +*/ diff --git a/delegator/delegator.admin.inc b/delegator/delegator.admin.inc index 600f03bc..30fff233 100644 --- a/delegator/delegator.admin.inc +++ b/delegator/delegator.admin.inc @@ -266,6 +266,10 @@ function delegator_admin_list_form(&$form_state) { '#value' => $title, ); + $form['handlers'][$id]['summary'] = array( + '#value' => delegator_get_handler_summary($plugin, $handler, $task, $form_state['subtask_id']), + ); + $form['handlers'][$id]['weight'] = array( '#type' => 'weight', '#default_value' => $info['weight'], @@ -441,7 +445,12 @@ function theme_delegator_admin_list_form($form) { 'class' => 'delegator-changed-col', ); - $title = theme('ctools_collapsible', drupal_render($element['title']), 'Imagine content here, please', TRUE); + if ($summary = drupal_render($element['summary'])) { + $title = theme('ctools_collapsible', drupal_render($element['title']), $summary, TRUE); + } + else { + $title = drupal_render($element['title']); + } $row[] = array( 'data' => $title, diff --git a/delegator/delegator.module b/delegator/delegator.module index 6a23cc16..938d6d8f 100644 --- a/delegator/delegator.module +++ b/delegator/delegator.module @@ -394,6 +394,15 @@ function delegator_get_handler_title($plugin, $handler, $task, $subtask_id) { } } +/** + * Get the admin summary (additional info) for a given handler. + */ +function delegator_get_handler_summary($plugin, $handler, $task, $subtask_id) { + if ($function = ctools_plugin_get_function($plugin, 'admin summary')) { + return $function($handler, $task, $subtask_id); + } +} + /** * Implementation of hook_ctools_plugin_dierctory() to let the system know * we implement task and task_handler plugins. diff --git a/delegator/help/api-task-handler.html b/delegator/help/api-task-handler.html index 72e6937b..ff45d873 100644 --- a/delegator/help/api-task-handler.html +++ b/delegator/help/api-task-handler.html @@ -4,7 +4,9 @@ task handler definition: description -- description of the task handler. task type -- The type of the task this handler can service. render -- callback of the function to render the handler. The arguments to this callback are specific to the task type. - title callback -- callback to render the admin title as this handler is listed. + admin title -- callback to render the admin title as this handler is listed. + params: $handler, $task, $subtask_id + admin summary -- callback to render what's in the collapsible info as the handler is listed. Optional. params: $handler, $task, $subtask_id default conf -- either an array() of default conf data or a callback that returns an array. params: $handler, $task, $subtask_id @@ -36,5 +38,7 @@ task handler definition: 'id2' => t('tab name'), ), + If a form name is blank it is a 'hidden' form -- it has no tab but can still be reached. + Notes: Because #required validation cannot be skipped when clicking cancel, please don't use it. \ No newline at end of file diff --git a/delegator/js/task-handlers.js b/delegator/js/task-handlers.js index 8460c226..4b666a32 100644 --- a/delegator/js/task-handlers.js +++ b/delegator/js/task-handlers.js @@ -38,4 +38,41 @@ Drupal.behaviors.zzGoLastDelegatorTaskList = function(context) { $next.trigger('click'); }); }); -} \ No newline at end of file +} + +Drupal.Delegator = {}; + +Drupal.Delegator.CollapsibleCallback = function($container, handle, content, toggle) { + var $parent = $container.parents('tr.draggable'); + var id = $parent.attr('id') + '-collapse'; + if (toggle.hasClass('ctools-toggle-collapsed')) { + // Force any other item to close, like an accordion: + $('#delegator-task-list-arrange .ctools-toggle:not(.ctools-toggle-collapsed)').trigger('click'); + // Closed, about to be opened. + var tr = '<tr class="delegator-collapsible" id="' + id + '"><td colspan=4></td></tr>'; + $parent.after(tr); + $('#' + id + ' td').append(content); + $('#' + id).addClass($parent.attr('class')); + } +}; + +Drupal.Delegator.CollapsibleCallbackAfterToggle = function($container, handle, content, toggle) { + var $parent = $container.parents('tr.draggable'); + var id = $parent.attr('id') + '-collapse'; + if (toggle.hasClass('ctools-toggle-collapsed')) { + // Was just closed. + content.hide(); + handle.after(content); + $('#' + id).remove(); + } +}; + +$(document).ready(function() { + Drupal.CTools.CollapsibleCallbacks.push(Drupal.Delegator.CollapsibleCallback); + Drupal.CTools.CollapsibleCallbacksAfterToggle.push(Drupal.Delegator.CollapsibleCallbackAfterToggle); + + // Force all our accordions to close when tabledragging to prevent ugliness: + $('#delegator-task-list-arrange .tabledrag-handle').mousedown(function() { + $('#delegator-task-list-arrange .ctools-toggle:not(.ctools-toggle-collapsed)').trigger('click'); + }); +}); \ No newline at end of file diff --git a/help/plugins-creating.html b/help/plugins-creating.html index b447dfdd..ac4db84f 100644 --- a/help/plugins-creating.html +++ b/help/plugins-creating.html @@ -25,4 +25,15 @@ Automatically filled in data: name module path - file \ No newline at end of file + file + + +General feature for callbacks: + either 'function_name' or + array( + 'file' => 'filename', + 'path' => 'filepath', // optional + 'function' => 'function_name' + ), + + Using ctools_plugin_get_function() of ctools_plugin_load_function() will take advantage. \ No newline at end of file diff --git a/includes/plugins.inc b/includes/plugins.inc index 4d9d09b6..d5b843eb 100644 --- a/includes/plugins.inc +++ b/includes/plugins.inc @@ -257,11 +257,30 @@ function _ctools_get_plugins($module, $type, $id = NULL) { function ctools_plugin_get_function($plugin, $function_name) { // If cached the .inc file may not have been loaded. require_once is quite safe // and fast so it's okay to keep calling it. - if ($plugin['file']) { + if (isset($plugin['file'])) { require_once './' . $plugin['path'] . '/' . $plugin['file']; } - if (isset($plugin[$function_name]) && function_exists($plugin[$function_name])) { - return $plugin[$function_name]; + + if (!isset($plugin[$function_name])) { + return; + } + + if (is_array($plugin[$function_name]) && isset($plugin[$function_name]['function'])) { + $function = $plugin[$function_name]['function']; + if (isset($plugin[$function_name]['file'])) { + $file = $plugin[$function_name]['file']; + if (isset($plugin[$function_name]['path'])) { + $file = $plugin[$function_name]['path'] . '/' . $file; + } + require_once './' . $file; + } + } + else { + $function = $plugin[$function_name]; + } + + if (function_exists($function)) { + return $function; } } diff --git a/js/collapsible-div.js b/js/collapsible-div.js index c453eb5f..a51035c1 100644 --- a/js/collapsible-div.js +++ b/js/collapsible-div.js @@ -21,6 +21,10 @@ if (!Drupal.CTools) { * a class, which will cause the container to draw collapsed. */ +// Set up an array for callbacks. +Drupal.CTools.CollapsibleCallbacks = []; +Drupal.CTools.CollapsibleCallbacksAfterToggle = []; + /** * Bind collapsible behavior to a given container. */ @@ -40,15 +44,27 @@ Drupal.CTools.bindCollapsible = function() { content.hide(); } - // Let both the toggle and the handle be clickable. - toggle.click(function() { - content.slideToggle(20); - toggle.toggleClass('ctools-toggle-collapsed'); - }); - handle.click(function() { - content.slideToggle(20); + var afterToggle = function() { + if (Drupal.CTools.CollapsibleCallbacksAfterToggle) { + for (i in Drupal.CTools.CollapsibleCallbacksAfterToggle) { + Drupal.CTools.CollapsibleCallbacksAfterToggle[i]($container, handle, content, toggle); + } + } + } + + var clickMe = function() { + if (Drupal.CTools.CollapsibleCallbacks) { + for (i in Drupal.CTools.CollapsibleCallbacks) { + Drupal.CTools.CollapsibleCallbacks[i]($container, handle, content, toggle); + } + } + content.slideToggle(100, afterToggle); toggle.toggleClass('ctools-toggle-collapsed'); - }); + } + + // Let both the toggle and the handle be clickable. + toggle.click(clickMe); + handle.click(clickMe); } }; -- GitLab