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