From 22f375ca452579a3882109846b6191d5f27593aa Mon Sep 17 00:00:00 2001 From: Earl Miles <merlin@logrus.com> Date: Sat, 18 Apr 2009 02:00:35 +0000 Subject: [PATCH] The rest of the content types! --- includes/content.inc | 76 ++++--- includes/content.menu.inc | 50 +++++ includes/content.theme.inc | 2 +- includes/context.inc | 2 - plugins/content_types/form/form.inc | 58 ++++++ plugins/content_types/form/icon_form.png | Bin 0 -> 460 bytes .../content_types/node_context/icon_node.png | Bin 0 -> 460 bytes .../node_context/node_attachments.inc | 45 +++++ .../node_context/node_book_nav.inc | 44 +++++ .../node_context/node_comment_form.inc | 72 +++++++ .../node_context/node_comments.inc | 172 ++++++++++++++++ .../node_context/node_content.inc | 185 ++++++++++++++++++ .../node_context/node_type_desc.inc | 48 +++++ .../node_form/icon_node_form.png | Bin 0 -> 460 bytes .../node_form/node_form_attachments.inc | 46 +++++ .../node_form/node_form_author.inc | 46 +++++ .../node_form/node_form_book.inc | 50 +++++ .../node_form/node_form_buttons.inc | 40 ++++ .../node_form/node_form_comment.inc | 46 +++++ .../node_form/node_form_input_format.inc | 44 +++++ .../content_types/node_form/node_form_log.inc | 33 ++++ .../node_form/node_form_menu.inc | 47 +++++ .../node_form/node_form_path.inc | 47 +++++ .../node_form/node_form_publishing.inc | 50 +++++ .../node_form/node_form_taxonomy.inc | 46 +++++ .../content_types/term_context/icon_term.png | Bin 0 -> 460 bytes .../term_context/term_description.inc | 51 +++++ .../content_types/term_context/term_list.inc | 114 +++++++++++ .../content_types/user_context/icon_user.png | Bin 0 -> 606 bytes .../user_context/profile_fields.inc | 132 +++++++++++++ .../user_context/profile_fields_pane.tpl.php | 17 ++ .../user_context/user_picture.inc | 43 ++++ .../user_context/user_profile.inc | 59 ++++++ .../vocabulary_context/icon_vocabulary.png | Bin 0 -> 460 bytes .../vocabulary_context/vocabulary_terms.inc | 96 +++++++++ 35 files changed, 1732 insertions(+), 29 deletions(-) create mode 100644 includes/content.menu.inc create mode 100644 plugins/content_types/form/form.inc create mode 100644 plugins/content_types/form/icon_form.png create mode 100644 plugins/content_types/node_context/icon_node.png create mode 100644 plugins/content_types/node_context/node_attachments.inc create mode 100644 plugins/content_types/node_context/node_book_nav.inc create mode 100644 plugins/content_types/node_context/node_comment_form.inc create mode 100644 plugins/content_types/node_context/node_comments.inc create mode 100644 plugins/content_types/node_context/node_content.inc create mode 100644 plugins/content_types/node_context/node_type_desc.inc create mode 100644 plugins/content_types/node_form/icon_node_form.png create mode 100644 plugins/content_types/node_form/node_form_attachments.inc create mode 100644 plugins/content_types/node_form/node_form_author.inc create mode 100644 plugins/content_types/node_form/node_form_book.inc create mode 100644 plugins/content_types/node_form/node_form_buttons.inc create mode 100644 plugins/content_types/node_form/node_form_comment.inc create mode 100644 plugins/content_types/node_form/node_form_input_format.inc create mode 100644 plugins/content_types/node_form/node_form_log.inc create mode 100644 plugins/content_types/node_form/node_form_menu.inc create mode 100644 plugins/content_types/node_form/node_form_path.inc create mode 100644 plugins/content_types/node_form/node_form_publishing.inc create mode 100644 plugins/content_types/node_form/node_form_taxonomy.inc create mode 100644 plugins/content_types/term_context/icon_term.png create mode 100644 plugins/content_types/term_context/term_description.inc create mode 100644 plugins/content_types/term_context/term_list.inc create mode 100644 plugins/content_types/user_context/icon_user.png create mode 100644 plugins/content_types/user_context/profile_fields.inc create mode 100644 plugins/content_types/user_context/profile_fields_pane.tpl.php create mode 100644 plugins/content_types/user_context/user_picture.inc create mode 100644 plugins/content_types/user_context/user_profile.inc create mode 100644 plugins/content_types/vocabulary_context/icon_vocabulary.png create mode 100644 plugins/content_types/vocabulary_context/vocabulary_terms.inc diff --git a/includes/content.inc b/includes/content.inc index 56063b73..c7909703 100644 --- a/includes/content.inc +++ b/includes/content.inc @@ -72,8 +72,8 @@ function ctools_content_defaults($info, &$plugin) { 'category' => $plugin['category'], ); - if (isset($plugin['required contexts'])) { - $type['required contexts'] = $plugin['required contexts']; + if (isset($plugin['required context'])) { + $type['required context'] = $plugin['required context']; } if (isset($plugin['top level'])) { $type['top level'] = $plugin['top level']; @@ -187,9 +187,16 @@ function ctools_content_get_subtype($type, $subtype_id) { if (isset($subtypes[$subtype_id])) { $subtype = $subtypes[$subtype_id]; } + // If there's only 1 and we somehow have the wrong subtype ID, do not + // care. Return the proper subtype anyway. + if (empty($subtype) && !empty($plugin['single'])) { + $subtype = current($subtypes); + } } - ctools_content_prepare_subtype($subtype, $plugin); + if ($subtype) { + ctools_content_prepare_subtype($subtype, $plugin); + } return $subtype; } @@ -247,15 +254,9 @@ function ctools_content_render($type, $subtype, $conf, $keywords = array(), $arg } if ($function = ctools_plugin_get_function($plugin, 'render callback')) { - if (empty($subtype['all contexts'])) { - $pane_context = ctools_content_select_context($type, $subtype, $conf, $context); - // FALSE means rendering was forbidden. - if ($pane_context === FALSE) { - return FALSE; - } - } - else { - $pane_context = $context; + $pane_context = ctools_content_select_context($plugin, $subtype, $conf, $context); + if ($pane_context === FALSE) { + return; } $content = $function($subtype, $conf, $args, $pane_context, $incoming_content); @@ -328,16 +329,11 @@ function ctools_content_admin_title($type, $subtype, $conf, $context = NULL) { } if ($function = ctools_plugin_get_function($plugin, 'admin title')) { - if (empty($subtype['all contexts'])) { - $pane_context = ctools_content_select_context($type, $subtype, $conf, $context); - // FALSE means rendering was forbidden. - if ($pane_context === FALSE) { - return FALSE; - } - } - else { - $pane_context = $context; + $pane_context = ctools_content_select_context($plugin, $subtype, $conf, $context); + if ($pane_context === FALSE) { + return t('@type:@subtype will not display due to missing context', array('@type' => $plugin['name'], '@subtype' => '@subtype')); } + return $function($subtype, $conf, $pane_context); } else if (isset($plugin['admin title'])) { @@ -435,8 +431,10 @@ function ctools_content_configure_form_defaults(&$form, &$form_state) { $contexts = isset($form_state['contexts']) ? $form_state['contexts'] : NULL; $conf = $form_state['conf']; + $add_submit = FALSE; if (!empty($subtype['required context']) && is_array($contexts)) { $form['context'] = ctools_context_selector($contexts, $subtype['required context'], isset($conf['context']) ? $conf['context'] : array()); + $add_submit = TRUE; } // Unless we're not allowed to override the title on this content type, add this @@ -467,11 +465,29 @@ function ctools_content_configure_form_defaults(&$form, &$form_state) { '#value' => t('You may use %keywords from contexts, as well as %title to contain the original title.'), ); } + $add_submit = TRUE; } + if ($add_submit) { + // '#submit' is already set up due to the wizard. + $form['#submit'][] = 'ctools_content_configure_form_defaults_submit'; + } return $form; } +/** + * Submit handler to store context/title override info. + */ +function ctools_content_configure_form_defaults_submit(&$form, &$form_state) { + if (isset($form_state['values']['context'])) { + $form_state['conf']['context'] = $form_state['values']['context']; + } + if (isset($form_state['values']['override_title'])) { + $form_state['conf']['override_title'] = $form_state['values']['override_title']; + $form_state['conf']['override_title_text'] = $form_state['values']['override_title_text']; + } +} + /** * Get the config form. * @@ -637,8 +653,8 @@ function ctools_content_get_all_types() { /** * Select the context to be used for a piece of content, based upon config. * - * @param $type - * The type of the content. + * @param $plugin + * The content plugin * @param $subtype * The subtype of the content. * @param $conf @@ -650,16 +666,24 @@ function ctools_content_get_all_types() { * The matching contexts or NULL if none or necessary, or FALSE if * requirements can't be met. */ -function ctools_content_select_context($type, $subtype, $conf, $contexts) { +function ctools_content_select_context($plugin, $subtype, $conf, $contexts) { // Identify which of our possible contexts apply. if (empty($subtype)) { return; } - $subtype_info = ctools_content_get_subtype($type, $subtype); + $subtype_info = ctools_content_get_subtype($plugin, $subtype); + if (empty($subtype_info)) { + return FALSE; + } + + if (!empty($subtype_info['all contexts']) || !empty($plugin['all contexts'])) { + return $contexts; + } + // If the content requires a context, fetch it; if no context is returned, // do not display the pane. - if (empty($subtype_info) || empty($subtype_info['required context'])) { + if (empty($subtype_info['required context'])) { return; } diff --git a/includes/content.menu.inc b/includes/content.menu.inc new file mode 100644 index 00000000..313f2c33 --- /dev/null +++ b/includes/content.menu.inc @@ -0,0 +1,50 @@ +<?php +// $Id$ + +/** + * @file + * Contains menu item registration for the content tool. + * + * The menu items registered are AJAX callbacks for the things like + * autocomplete and other tools needed by the content types. + */ + +function ctools_content_menu(&$items) { + $base = array( + 'access arguments' => array('access content'), + 'type' => MENU_CALLBACK, + 'file' => 'includes/content.menu.inc', + ); + $items['ctools/autocomplete/node'] = array( + 'page callback' => 'ctools_content_autocomplete_node', + ) + $base; +} + +/** + * Helper function for autocompletion of node titles. + */ +function ctools_content_autocomplete_node($string) { + if ($string != '') { + $preg_matches = array(); + $match = preg_match('/\[nid: (\d+)\]/', $string, $preg_matches); + if (!$match) { + $match = preg_match('/^nid: (\d+)/', $string, $preg_matches); + } + if ($match) { + $arg = $preg_matches[1]; + $where = "n.nid = %d"; + } + else { + $arg = $string; + $where = "LOWER(title) LIKE LOWER('%%%s%%')"; + } + $result = db_query_range(db_rewrite_sql("SELECT n.nid, n. title, u.name FROM {node} n INNER JOIN {users} u ON u.uid = n.uid WHERE $where"), $arg, 0, 10); + + $matches = array(); + while ($node = db_fetch_object($result)) { + $name = empty($node->name) ? variable_get('anonymous', t('Anonymous')) : check_plain($node->name); + $matches[$node->title . " [nid: $node->nid]"] = '<span class="autocomplete_title">' . check_plain($node->title) . '</span> <span class="autocomplete_user">(' . t('by @user', array('@user' => $name)) . ')</span>'; + } + drupal_json($matches); + } +} diff --git a/includes/content.theme.inc b/includes/content.theme.inc index 14d716c3..2f82cf2b 100644 --- a/includes/content.theme.inc +++ b/includes/content.theme.inc @@ -16,7 +16,7 @@ function ctools_content_theme(&$theme) { $plugins = ctools_get_content_types(); foreach ($plugins as $plugin) { if (ctools_plugin_get_function($plugin, 'hook theme')) { - $function($theme); + $function($theme, $plugin); } } } diff --git a/includes/context.inc b/includes/context.inc index e5a05a51..a8cca357 100644 --- a/includes/context.inc +++ b/includes/context.inc @@ -255,8 +255,6 @@ function ctools_context_selector($contexts, $required, $default) { function _ctools_context_selector($contexts, $required, $default, $num = 0) { $filtered = ctools_context_filter($contexts, $required); - dsm($required); - dsm($filtered); $count = count($filtered); $form = array(); diff --git a/plugins/content_types/form/form.inc b/plugins/content_types/form/form.inc new file mode 100644 index 00000000..3f808f9c --- /dev/null +++ b/plugins/content_types/form/form.inc @@ -0,0 +1,58 @@ +<?php +// $Id$ + + +/** + * Callback function to supply a list of content types. + */ +function ctools_form_ctools_content_types() { + return array( + // only provides a single content type + 'single' => TRUE, + 'render last' => TRUE, + 'title' => t('General form'), + 'icon' => 'icon_form.png', + 'description' => t('Everything in the form that is not displayed by other content.'), + 'required context' => new ctools_context_required(t('Form'), 'form'), + 'category' => t('Form'), + ); +} + +/** + * Output function for the 'node' content type. Outputs a node + * based on the module and delta supplied in the configuration. + */ +function ctools_form_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = 'form'; + + if (isset($context->form)) { + $block->title = $context->form_title; + if (!empty($context->form_id)) { + // If this is a form, drupal_render it. + $block->content = drupal_render($context->form); + } + else { + // Otherwise just spit back what we were given. This is probably an + // error message or something. + $block->content = $context->form; + } + $block->delta = $context->form_id; + } + else { + $block->title = t('Form'); + $block->content = t('Form goes here.'); + $block->delta = 'unknown'; + } + + return $block; +} + +function ctools_form_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" base form', array('@s' => $context->identifier)); +} + +function ctools_form_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to override title + // and stuff. +} diff --git a/plugins/content_types/form/icon_form.png b/plugins/content_types/form/icon_form.png new file mode 100644 index 0000000000000000000000000000000000000000..f0417cb6515aa7fb2856c90722b386ed99bfc501 GIT binary patch literal 460 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!63?wyl`GbKJOS+@4BLl;=8&~`zjDUQ}64!{5 z;QX|b^2DN42FH~Aq*MjZ+{E<Mpwz^a%EFVWHb6yD0X`wFE+MINE`9S#t2456o^<}3 zj-i#Dir(tW_tssyv-H%p2QS{7I(h2a%eT+pe}4bz%ad1cEnR}A?LYhN=dY$Udly~( zcH;T(jkkW}PF%M8!OyMteyqCr{ovzY|Ns978vk!K_g5e#R}$nG3>3i)7~V+8!~s>F z^K@|xskoK&Ao2MF5fKqKrozHP$H>SAhNOh)j!aFDHZd?BIiRS_#db+SLGK)kfCJAQ zqeTU>kA$3ffHDdi%#){0m?9#a#K57jfnf&sBX3Sa1{Q(l4Luqi?2o!UJvli+Tm$)4 z3=Ry8tOs@-j1d%!V{BkJBmmUGR=9zogZqi`CnIJS0S|@~)0;j>{+!9k)ZrMzp`^g{ zWb=lCjZ7>85e%#3G!7i+5^!)}RE&#@t221;=y4h|P`{c&LV^GTQ(9VBm>F9G12cn9 XiE#1KH6NpaK4b87^>bP0l+XkKLQcLN literal 0 HcmV?d00001 diff --git a/plugins/content_types/node_context/icon_node.png b/plugins/content_types/node_context/icon_node.png new file mode 100644 index 0000000000000000000000000000000000000000..f0417cb6515aa7fb2856c90722b386ed99bfc501 GIT binary patch literal 460 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!63?wyl`GbKJOS+@4BLl;=8&~`zjDUQ}64!{5 z;QX|b^2DN42FH~Aq*MjZ+{E<Mpwz^a%EFVWHb6yD0X`wFE+MINE`9S#t2456o^<}3 zj-i#Dir(tW_tssyv-H%p2QS{7I(h2a%eT+pe}4bz%ad1cEnR}A?LYhN=dY$Udly~( zcH;T(jkkW}PF%M8!OyMteyqCr{ovzY|Ns978vk!K_g5e#R}$nG3>3i)7~V+8!~s>F z^K@|xskoK&Ao2MF5fKqKrozHP$H>SAhNOh)j!aFDHZd?BIiRS_#db+SLGK)kfCJAQ zqeTU>kA$3ffHDdi%#){0m?9#a#K57jfnf&sBX3Sa1{Q(l4Luqi?2o!UJvli+Tm$)4 z3=Ry8tOs@-j1d%!V{BkJBmmUGR=9zogZqi`CnIJS0S|@~)0;j>{+!9k)ZrMzp`^g{ zWb=lCjZ7>85e%#3G!7i+5^!)}RE&#@t221;=y4h|P`{c&LV^GTQ(9VBm>F9G12cn9 XiE#1KH6NpaK4b87^>bP0l+XkKLQcLN literal 0 HcmV?d00001 diff --git a/plugins/content_types/node_context/node_attachments.inc b/plugins/content_types/node_context/node_attachments.inc new file mode 100644 index 00000000..bb80c0a7 --- /dev/null +++ b/plugins/content_types/node_context/node_attachments.inc @@ -0,0 +1,45 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_attachments_ctools_content_types() { + return array( + 'single' => TRUE, + 'title' => t('Attached files'), + 'icon' => 'icon_node.png', + 'description' => t('A list of files attached to the node.'), + 'required context' => new ctools_context_required(t('Node'), 'node'), + 'category' => t('Node'), + ); +} + +function ctools_node_attachments_content_type_render($subtype, $conf, $panel_args, $context) { + $node = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'attachments'; + + $block->title = t('Attached files'); + if ($node) { + if (!empty($node->files)) { + $block->content = theme('upload_attachments', $node->files); + } + $block->delta = $node->nid; + } + else { + $block->content = t('Attached files go here.'); + $block->delta = 'unknown'; + } + + return $block; +} + +function ctools_node_attachments_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" attachments', array('@s' => $context->identifier)); +} + +function ctools_node_attachments_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} + diff --git a/plugins/content_types/node_context/node_book_nav.inc b/plugins/content_types/node_context/node_book_nav.inc new file mode 100644 index 00000000..200d0f2c --- /dev/null +++ b/plugins/content_types/node_context/node_book_nav.inc @@ -0,0 +1,44 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_book_nav_ctools_content_types() { + if (module_exists('book')) { + return array( + 'single' => TRUE, + 'title' => t('Book navigation'), + 'icon' => 'icon_node.png', + 'description' => t('The navigation menu the book the node belongs to.'), + 'required context' => new ctools_context_required(t('Node'), 'node'), + 'category' => t('Node'), + ); + } +} + +function ctools_node_book_nav_content_type_render($subtype, $conf, $panel_args, $context) { + $node = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'book_nav'; + + $block->title = t('Book navigation'); + if ($node) { + $block->content = isset($node->book) ? theme('book_navigation', $node->book) : ''; + $block->delta = $node->nid; + } + else { + $block->content = t('Book navigation goes here.'); + $block->delta = 'unknown'; + } + + return $block; +} + +function ctools_node_book_nav_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" book navigation', array('@s' => $context->identifier)); +} + +function ctools_node_book_nav_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_context/node_comment_form.inc b/plugins/content_types/node_context/node_comment_form.inc new file mode 100644 index 00000000..2d7ffe64 --- /dev/null +++ b/plugins/content_types/node_context/node_comment_form.inc @@ -0,0 +1,72 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_comment_form_ctools_content_types() { + if (module_exists('comment')) { + return array( + 'single' => TRUE, + 'title' => t('Comment form'), + 'icon' => 'icon_node.png', + 'description' => t('A form to add a new comment.'), + 'required context' => new ctools_context_required(t('Node'), 'node'), + 'category' => t('Node'), + ); + } +} + +function ctools_node_comment_form_content_type_render($subtype, $conf, $panel_args, $context) { + $node = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'comments'; + $block->delta = $node->nid; + + $block->title = t('Add comment'); + + if (empty($node)) { + $block->content = t('Comment form here.'); + } + else { + if (user_access('post comments') && node_comment_mode($node->nid) == COMMENT_NODE_READ_WRITE) { + ctools_include('form'); + $form_state = array( + 'ctools comment alter' => TRUE, + 'node' => $node, + 'args' => array(array('nid' => $node->nid)) + ); + $block->content = ctools_build_form('comment_form', $form_state); + } + } + + return $block; +} + +function ctools_node_comment_form_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" comment form', array('@s' => $context->identifier)); +} + +function ctools_node_comment_form_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} + +/** + * Alter the comment form to get a little more control over it. + */ +function ctools_form_comment_form_alter(&$form, &$form_state) { + if (!empty($form_state['ctools comment alter'])) { + $node = + // force the form to post back to wherever we are. + $url = parse_url($_GET['q']); + $form['#action'] = url($url['path'], array('fragment' => 'comment-form')); + if (empty($form['#submit'])) { + $form['#submit'] = array('comment_form_submit'); + } + $form['#submit'][] = 'ctools_node_comment_form_submit'; + } +} + +function ctools_node_comment_form_submit(&$form, &$form_state) { + $form_state['redirect'][0] = $_GET['q']; +} diff --git a/plugins/content_types/node_context/node_comments.inc b/plugins/content_types/node_context/node_comments.inc new file mode 100644 index 00000000..c261426b --- /dev/null +++ b/plugins/content_types/node_context/node_comments.inc @@ -0,0 +1,172 @@ +<?php +// $Id$ + +/** + * Plugin declaration function - returns a plugin definition array that + * describes the content type. + */ +function ctools_node_comments_ctools_content_types() { + if (module_exists('comment')) { + return array( + 'single' => TRUE, + 'title' => t('Node comments'), + 'icon' => 'icon_node.png', + 'description' => t('The comments of the referenced node.'), + 'required context' => new ctools_context_required(t('Node'), 'node'), + 'category' => t('Node'), + 'defaults' => array( + 'mode' => variable_get('comment_default_mode', COMMENT_MODE_THREADED_EXPANDED), + 'order' => variable_get('comment_default_order', COMMENT_ORDER_NEWEST_FIRST), + 'comments_per_page' => variable_get('comment_default_per_page', '50'), + ), + ); + } +} + +function ctools_node_comments_content_type_render($subtype, $conf, $panel_args, $context) { + $node = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'comments'; + $block->delta = $node->nid; + + $block->title = t('Comments'); + if (empty($node)) { + $block->content = t('Node comments go here.'); + } + else { + $block->content = ctools_comment_render($node, $conf); + // Update the history table, stating that this user viewed this node. + node_tag_new($node->nid); + } + + return $block; +} + +function ctools_node_comments_content_type_edit_form(&$form, &$form_state) { + $conf = $form_state['conf']; + $form['mode'] = array( + '#type' => 'select', + '#title' => t('Mode'), + '#default_value' => $conf['mode'], + '#options' => _comment_get_modes(), + '#weight' => 1, + ); + $form['order'] = array( + '#type' => 'select', + '#title' => t('Sort'), + '#default_value' => $conf['order'], + '#options' => _comment_get_orders(), + '#weight' => 2, + ); + foreach (_comment_per_page() as $i) { + $options[$i] = t('!a comments per page', array('!a' => $i)); + } + $form['comments_per_page'] = array('#type' => 'select', + '#title' => t('Pager'), + '#default_value' => $conf['comments_per_page'], + '#options' => $options, + '#weight' => 3, + ); +} + +function ctools_node_comments_content_type_edit_form_submit(&$form, &$form_state) { + // Copy everything from our defaults. + foreach (array_keys($form_state['plugin']['defaults']) as $key) { + $form_state['conf'][$key] = $form_state['values'][$key]; + } +} + +function ctools_node_comments_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" comments', array('@s' => $context->identifier)); +} + +/** + * This function is a somewhat stripped down version of comment_render + * that removes a bunch of cruft that we both don't need, and makes it + * difficult to modify this. + */ +function ctools_comment_render($node, $conf) { + $output = ''; + if (!user_access('access comments')) { + return; + } + + $mode = $conf['mode']; + $order = $conf['order']; + $comments_per_page = $conf['comments_per_page']; + + // Multiple comment view + $query_count = 'SELECT COUNT(*) FROM {comments} WHERE nid = %d'; + $query = 'SELECT c.cid as cid, c.pid, c.nid, c.subject, c.comment, c.format, c.timestamp, c.name, c.mail, c.homepage, u.uid, u.name AS registered_name, u.picture, u.data, c.thread, c.status FROM {comments} c INNER JOIN {users} u ON c.uid = u.uid WHERE c.nid = %d'; + + $query_args = array($node->nid); + if (!user_access('administer comments')) { + $query .= ' AND c.status = %d'; + $query_count .= ' AND status = %d'; + $query_args[] = COMMENT_PUBLISHED; + } + + if ($order == COMMENT_ORDER_NEWEST_FIRST) { + if ($mode == COMMENT_MODE_FLAT_COLLAPSED || $mode == COMMENT_MODE_FLAT_EXPANDED) { + $query .= ' ORDER BY c.timestamp DESC'; + } + else { + $query .= ' ORDER BY c.thread DESC'; + } + } + else if ($order == COMMENT_ORDER_OLDEST_FIRST) { + if ($mode == COMMENT_MODE_FLAT_COLLAPSED || $mode == COMMENT_MODE_FLAT_EXPANDED) { + $query .= ' ORDER BY c.timestamp'; + } + else { + $query .= ' ORDER BY SUBSTRING(c.thread, 1, (LENGTH(c.thread) - 1))'; + } + } + + // Start a form, for use with comment control. + $result = pager_query($query, $comments_per_page, 0, $query_count, $query_args); + + $divs = 0; + $last_depth = 0; + drupal_add_css(drupal_get_path('module', 'comment') .'/comment.css'); + while ($comment = db_fetch_object($result)) { + $comment = drupal_unpack($comment); + $comment->name = $comment->uid ? $comment->registered_name : $comment->name; + $comment->depth = count(explode('.', $comment->thread)) - 1; + + if ($mode == COMMENT_MODE_THREADED_COLLAPSED || $mode == COMMENT_MODE_THREADED_EXPANDED) { + if ($comment->depth > $last_depth) { + $divs++; + $output .= '<div class="indented">'; + $last_depth++; + } + else { + while ($comment->depth < $last_depth) { + $divs--; + $output .= '</div>'; + $last_depth--; + } + } + } + + if ($mode == COMMENT_MODE_FLAT_COLLAPSED) { + $output .= theme('comment_flat_collapsed', $comment, $node); + } + else if ($mode == COMMENT_MODE_FLAT_EXPANDED) { + $output .= theme('comment_flat_expanded', $comment, $node); + } + else if ($mode == COMMENT_MODE_THREADED_COLLAPSED) { + $output .= theme('comment_thread_collapsed', $comment, $node); + } + else if ($mode == COMMENT_MODE_THREADED_EXPANDED) { + $output .= theme('comment_thread_expanded', $comment, $node); + } + } + for ($i = 0; $i < $divs; $i++) { + $output .= '</div>'; + } + $output .= theme('pager', NULL, $comments_per_page, 0); + + return $output; +} + diff --git a/plugins/content_types/node_context/node_content.inc b/plugins/content_types/node_context/node_content.inc new file mode 100644 index 00000000..2aec36f9 --- /dev/null +++ b/plugins/content_types/node_context/node_content.inc @@ -0,0 +1,185 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function ctools_node_content_ctools_content_types() { + $items['node_content'] = array( + 'single' => TRUE, + 'title' => t('Node content'), + 'icon' => 'icon_node.png', + 'description' => t('The content of the referenced node.'), + 'required context' => new ctools_context_required(t('Node'), 'node'), + 'category' => t('Node'), + 'defaults' => array( + 'links' => TRUE, + 'page' => TRUE, + 'no_extras' => TRUE, + 'override_title' => FALSE, + 'override_title_text' => '', + 'teaser' => TRUE, + 'identifier' => '', + 'link' => TRUE, + 'leave_node_title' => FALSE, + ), + ); + return $items; +} + +/** + * Render the node content. + */ +function ctools_node_content_content_type_render($subtype, $conf, $panel_args, $context) { + if (!empty($context) && empty($context->data)) { + return; + } + + $node = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'node'; + $block->delta = $node->nid; + + if (empty($node)) { + $block->delta = 'placeholder'; + $block->title = t('Node title.'); + $block->content = t('Node content goes here.'); + } + else { + if (!empty($conf['identifier'])) { + $node->panel_identifier = $conf['identifier']; + } + + $block->title = $node->title; + if (empty($conf['leave_node_title'])) { + $node->title = NULL; + } + $block->content = ctools_node_content_render_node($node, $conf); + } + + if (node_access('update', $node)) { + $block->admin_links['update'] = array( + 'title' => t('Edit node'), + 'alt' => t("Edit this node"), + 'href' => "node/$node->nid/edit", + 'query' => drupal_get_destination(), + ); + } + + if (!empty($conf['link']) && $node) { + $block->title_link = "node/$node->nid"; + } + + return $block; +} + +function ctools_node_content_render_node($node, $conf) { + // The build mode identifies the target for which the node is built. + if (!isset($node->build_mode)) { + $node->build_mode = NODE_BUILD_NORMAL; + } + + // Remove the delimiter (if any) that separates the teaser from the body. + $node->body = str_replace('<!--break-->', '', $node->body); + + // The 'view' hook can be implemented to overwrite the default function + // to display nodes. + if (node_hook($node, 'view')) { + $node = node_invoke($node, 'view', $conf['teaser'], $conf['page']); + } + else { + $node = node_prepare($node, $conf['teaser']); + } + + if (empty($conf['no_extras'])) { + // Allow modules to make their own additions to the node. + node_invoke_nodeapi($node, 'view', $conf['teaser'], $conf['page']); + } + + if ($conf['links']) { + $node->links = module_invoke_all('link', 'node', $node, $conf['teaser']); + drupal_alter('link', $node->links, $node); + } + + // Set the proper node part, then unset unused $node part so that a bad + // theme can not open a security hole. + $content = drupal_render($node->content); + if ($conf['teaser']) { + $node->teaser = $content; + unset($node->body); + } + else { + $node->body = $content; + unset($node->teaser); + } + + // Allow modules to modify the fully-built node. + node_invoke_nodeapi($node, 'alter', $conf['teaser'], $conf['page']); + + return theme('node', $node, $conf['teaser'], $conf['page']); +} + +/** + * Returns an edit form for the custom type. + */ +function ctools_node_content_content_type_edit_form(&$form, &$form_state) { + $form['link'] = array( + '#title' => t('Link title to node'), + '#type' => 'checkbox', + '#default_value' => $conf['link'], + '#description' => t('Check here to make the title link to the node.'), + ); + $form['teaser'] = array( + '#title' => t('Teaser'), + '#type' => 'checkbox', + '#default_value' => $conf['teaser'], + '#description' => t('Check here to show only the node teaser.'), + ); + $form['page'] = array( + '#type' => 'checkbox', + '#default_value' => $conf['page'], + '#title' => t('Node page'), + '#description' => t('Check here if the node is being displayed on a page by itself.'), + ); + $form['links'] = array( + '#type' => 'checkbox', + '#default_value' => $conf['links'], + '#title' => t('Display links'), + '#description' => t('Check here to display the links with the post.'), + ); + + $form['no_extras'] = array( + '#type' => 'checkbox', + '#default_value' => $conf['no_extras'], + '#title' => t('No extras'), + '#description' => t('Check here to disable additions that modules might make to the node, such as file attachments and CCK fields; this should just display the basic teaser or body.'), + ); + + $form['identifier'] = array( + '#type' => 'textfield', + '#default_value' => $conf['identifier'], + '#title' => t('Identifier'), + '#description' => t('Whatever is placed here will appear in $node->panel_identifier in the node template to make it easier to theme a node or part of a node as necessary. This identifier will automatically be added as a node template suggestion: node-panel-IDENTIFIER.tpl.php'), + ); + + $form['leave_node_title'] = array( + '#type' => 'checkbox', + '#default_value' => $conf['leave_node_title'], + '#title' => t('Leave node title'), + '#description' => t('Advanced: if checked, do not touch the node title; this can cause the node title to appear twice unless your theme is aware of this.'), + ); + + return $form; +} + +function ctools_node_content_content_type_edit_form_submit(&$form, &$form_state) { + // Copy everything from our defaults. + foreach (array_keys($form_state['plugin']['defaults']) as $key) { + $form_state['conf'][$key] = $form_state['values'][$key]; + } +} + +function ctools_node_content_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" content', array('@s' => $context->identifier)); +} + diff --git a/plugins/content_types/node_context/node_type_desc.inc b/plugins/content_types/node_context/node_type_desc.inc new file mode 100644 index 00000000..795b0eb1 --- /dev/null +++ b/plugins/content_types/node_context/node_type_desc.inc @@ -0,0 +1,48 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function ctools_node_type_desc_ctools_content_types() { + return array( + 'single' => TRUE, + 'title' => t('Node type description'), + 'icon' => 'icon_node.png', + 'description' => t('Node type description.'), + 'required context' => new ctools_context_required(t('Node'), 'node'), + 'category' => t('Node'), + ); +} + +/** + * Output function for the 'node' content type. Outputs a node + * based on the module and delta supplied in the configuration. + */ +function ctools_node_type_desc_content_type_render($subtype, $conf, $panel_args, $context) { + $node = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'node_type'; + + if ($node) { + $type = node_get_types('type', $node); + $block->title = $type->name; + $block->content = filter_xss_admin($type->description); + $block->delta = $node->type; + } + else { + $block->title = t('Node type description'); + $block->content = t('Node type description goes here.'); + $block->delta = 'unknown'; + } + + return $block; +} + +function ctools_node_type_desc_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" type description', array('@s' => $context->identifier)); +} + +function ctools_node_type_desc_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/icon_node_form.png b/plugins/content_types/node_form/icon_node_form.png new file mode 100644 index 0000000000000000000000000000000000000000..f0417cb6515aa7fb2856c90722b386ed99bfc501 GIT binary patch literal 460 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!63?wyl`GbKJOS+@4BLl;=8&~`zjDUQ}64!{5 z;QX|b^2DN42FH~Aq*MjZ+{E<Mpwz^a%EFVWHb6yD0X`wFE+MINE`9S#t2456o^<}3 zj-i#Dir(tW_tssyv-H%p2QS{7I(h2a%eT+pe}4bz%ad1cEnR}A?LYhN=dY$Udly~( zcH;T(jkkW}PF%M8!OyMteyqCr{ovzY|Ns978vk!K_g5e#R}$nG3>3i)7~V+8!~s>F z^K@|xskoK&Ao2MF5fKqKrozHP$H>SAhNOh)j!aFDHZd?BIiRS_#db+SLGK)kfCJAQ zqeTU>kA$3ffHDdi%#){0m?9#a#K57jfnf&sBX3Sa1{Q(l4Luqi?2o!UJvli+Tm$)4 z3=Ry8tOs@-j1d%!V{BkJBmmUGR=9zogZqi`CnIJS0S|@~)0;j>{+!9k)ZrMzp`^g{ zWb=lCjZ7>85e%#3G!7i+5^!)}RE&#@t221;=y4h|P`{c&LV^GTQ(9VBm>F9G12cn9 XiE#1KH6NpaK4b87^>bP0l+XkKLQcLN literal 0 HcmV?d00001 diff --git a/plugins/content_types/node_form/node_form_attachments.inc b/plugins/content_types/node_form/node_form_attachments.inc new file mode 100644 index 00000000..2d614b0b --- /dev/null +++ b/plugins/content_types/node_form/node_form_attachments.inc @@ -0,0 +1,46 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_attachments_ctools_content_types() { + if (module_exists('upload')) { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form file attachments'), + 'description' => t('File attachments on the Node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); + } +} + +function ctools_node_form_attachments_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('Attach files'); + $block->delta = 'url-path-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['attachments']['#access'])) { + // remove the fieldset + unset($context->form['attachments']['#type']); + $block->content = drupal_render($context->form['attachments']); + } + } + else { + $block->content = t('Attach files.'); + } + return $block; +} + +function ctools_node_form_attachments_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form attach files', array('@s' => $context->identifier)); +} + +function ctools_node_form_attachments_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_author.inc b/plugins/content_types/node_form/node_form_author.inc new file mode 100644 index 00000000..4885cba6 --- /dev/null +++ b/plugins/content_types/node_form/node_form_author.inc @@ -0,0 +1,46 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_author_ctools_content_types() { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form author information'), + 'description' => t('Author information on the Node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); +} + +function ctools_node_form_author_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = $block->content = t('Authoring information'); + $block->delta = 'author-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['author']['#access'])) { + // remove the fieldset + unset($context->form['author']['#type']); + $context->form['author']['name']['#size'] /= 2; + $context->form['author']['date']['#size'] /= 2; + $block->content = drupal_render($context->form['author']); + } + } + else { + $block->content = t('Authoring information.'); + } + return $block; +} + +function ctools_node_form_author_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form publishing options', array('@s' => $context->identifier)); +} + +function ctools_node_form_author_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_book.inc b/plugins/content_types/node_form/node_form_book.inc new file mode 100644 index 00000000..a2a878c2 --- /dev/null +++ b/plugins/content_types/node_form/node_form_book.inc @@ -0,0 +1,50 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_book_ctools_content_types() { + if (module_exists('book')) { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form book options'), + 'description' => t('Book options for the node.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); + } +} + +function ctools_node_form_book_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('Book options'); + $block->delta = 'book-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id)) { + $block->content = ''; + if ($context->form['parent']['#type'] != 'value') { + $block->content .= drupal_render($context->form['parent']); + } + if ($context->form['weight']['#type'] != 'value') { + $block->content .= drupal_render($context->form['weight']); + } + } + } + else { + $block->content = t('Book options.'); + } + return $block; +} + +function ctools_node_form_book_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form book options', array('@s' => $context->identifier)); +} + +function ctools_node_form_book_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_buttons.inc b/plugins/content_types/node_form/node_form_buttons.inc new file mode 100644 index 00000000..334670fc --- /dev/null +++ b/plugins/content_types/node_form/node_form_buttons.inc @@ -0,0 +1,40 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_buttons_ctools_content_types() { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form submit buttons'), + 'description' => t('Submit buttons for the node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); +} + +function ctools_node_form_buttons_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = ''; + $block->delta = 'buttons'; + + if (isset($context->form)) { + $block->content = drupal_render($context->form['buttons']); + } + else { + $block->content = t('Node form buttons.'); + } + return $block; +} + +function ctools_node_form_buttons_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form submit buttons', array('@s' => $context->identifier)); +} + +function ctools_node_form_buttons_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_comment.inc b/plugins/content_types/node_form/node_form_comment.inc new file mode 100644 index 00000000..6d41c8a4 --- /dev/null +++ b/plugins/content_types/node_form/node_form_comment.inc @@ -0,0 +1,46 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_comment_ctools_content_types() { + if (module_exists('comment')) { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form comment settings'), + 'description' => t('Comment settings on the Node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); + } +} + +function ctools_node_form_comment_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('Comment options'); + $block->delta = 'comment-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['comment_settings']['#access'])) { + // remove the fieldset + unset($context->form['comment_settings']['#type']); + $block->content = drupal_render($context->form['comment_settings']); + } + } + else { + $block->content = t('Comment options.'); + } + return $block; +} + +function ctools_node_form_comment_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form comment settings', array('@s' => $context->identifier)); +} + +function ctools_node_form_comment_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_input_format.inc b/plugins/content_types/node_form/node_form_input_format.inc new file mode 100644 index 00000000..c7b51d53 --- /dev/null +++ b/plugins/content_types/node_form/node_form_input_format.inc @@ -0,0 +1,44 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_input_format_ctools_content_types() { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form input format'), + 'description' => t('Input format for the body field on a node.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); +} + +function ctools_node_form_input_format_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('Input format'); + $block->delta = 'format-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['body_filter']['format'])) { + // remove the fieldset + unset($context->form['body_filter']['format']['#type']); + $block->content = drupal_render($context->form['body_filter']['format']); + } + } + else { + $block->content = t('Input format.'); + } + return $block; +} + +function ctools_node_form_input_format_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form input format', array('@s' => $context->identifier)); +} + +function ctools_node_form_input_format_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_log.inc b/plugins/content_types/node_form/node_form_log.inc new file mode 100644 index 00000000..e1ec0390 --- /dev/null +++ b/plugins/content_types/node_form/node_form_log.inc @@ -0,0 +1,33 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_log_ctools_content_types() { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form revision log message'), + 'description' => t('Revision log message for the node.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); +} + +function ctools_node_form_log_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + // @todo -- this was never implemented!? + + return $block; +} + +function ctools_node_form_log_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form revision log', array('@s' => $context->identifier)); +} + +function ctools_node_form_log_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_menu.inc b/plugins/content_types/node_form/node_form_menu.inc new file mode 100644 index 00000000..eb6fa6d8 --- /dev/null +++ b/plugins/content_types/node_form/node_form_menu.inc @@ -0,0 +1,47 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_menu_ctools_content_types() { + if (module_exists('menu')) { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form menu settings'), + 'description' => t('Menu settings on the Node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); + } +} + +function ctools_node_form_menu_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('Menu options'); + $block->delta = 'menu-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['menu']['#access'])) { + // remove the fieldset + unset($context->form['menu']['#type']); + $context->form['menu']['link_title']['#size'] /= 2; + $block->content = drupal_render($context->form['menu']); + } + } + else { + $block->content = t('Menu options.'); + } + return $block; +} + +function ctools_node_form_menu_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form menu settings', array('@s' => $context->identifier)); +} + +function ctools_node_form_menu_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_path.inc b/plugins/content_types/node_form/node_form_path.inc new file mode 100644 index 00000000..ef16b272 --- /dev/null +++ b/plugins/content_types/node_form/node_form_path.inc @@ -0,0 +1,47 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_path_ctools_content_types() { + if (module_exists('path')) { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form url path settings'), + 'description' => t('Publishing options on the Node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); + } +} + +function ctools_node_form_path_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('URL path options'); + $block->delta = 'url-path-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['path']['#access'])) { + // remove the fieldset + unset($context->form['path']['#type']); + $context->form['path']['path']['#size'] /= 2; + $block->content = drupal_render($context->form['path']); + } + } + else { + $block->content = t('URL Path options.'); + } + return $block; +} + +function ctools_node_form_path_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form path options', array('@s' => $context->identifier)); +} + +function ctools_node_form_path_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_publishing.inc b/plugins/content_types/node_form/node_form_publishing.inc new file mode 100644 index 00000000..c43a1f67 --- /dev/null +++ b/plugins/content_types/node_form/node_form_publishing.inc @@ -0,0 +1,50 @@ +<?php +// $Id$ + +/** + * @file + * Publishing options form for the node. This contains the basic settings + * like published, moderated, node revision, etc. + */ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_publishing_ctools_content_types() { + return array( + 'single' => TRUE, + 'title' => t('Node form publishing options'), + 'icon' => 'icon_node_form.png', + 'description' => t('Publishing options on the Node form.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); +} + +function ctools_node_form_publishing_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + + $block->title = t('Publishing options'); + $block->module = t('node_form'); + $block->delta = 'publishing-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && $context->form['options']['#type'] == 'fieldset') { + // remove the fieldset + unset($context->form['options']['#type']); + $block->content = drupal_render($context->form['options']); + } + } + else { + $block->content = t('Publishing options.'); + } + return $block; +} + +function ctools_node_form_publishing_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form author information', array('@s' => $context->identifier)); +} + +function ctools_node_form_publishing_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/node_form/node_form_taxonomy.inc b/plugins/content_types/node_form/node_form_taxonomy.inc new file mode 100644 index 00000000..3fd62199 --- /dev/null +++ b/plugins/content_types/node_form/node_form_taxonomy.inc @@ -0,0 +1,46 @@ +<?php +// $Id$ + +/** + * Implementation of specially named hook_ctools_content_types() + */ +function ctools_node_form_taxonomy_ctools_content_types() { + if (module_exists('taxonomy')) { + return array( + 'single' => TRUE, + 'icon' => 'icon_node_form.png', + 'title' => t('Node form categories'), + 'description' => t('Taxonomy categories for the node.'), + 'required context' => new ctools_context_required(t('Form'), 'node_form'), + 'category' => t('Form'), + ); + } +} + +function ctools_node_form_taxonomy_content_type_render($subtype, $conf, $panel_args, &$context) { + $block = new stdClass(); + $block->module = t('node_form'); + + $block->title = t('Categories'); + $block->delta = 'url-path-options'; + + if (isset($context->form)) { + if (!empty($context->form->form_id) && !empty($context->form['taxonomy'])) { + // remove the fieldset + unset($context->form['taxonomy']['#type']); + $block->content = drupal_render($context->form['taxonomy']); + } + } + else { + $block->content = t('Categories.'); + } + return $block; +} + +function ctools_node_form_taxonomy_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" node form select taxonomy', array('@s' => $context->identifier)); +} + +function ctools_node_form_taxonomy_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/term_context/icon_term.png b/plugins/content_types/term_context/icon_term.png new file mode 100644 index 0000000000000000000000000000000000000000..f0417cb6515aa7fb2856c90722b386ed99bfc501 GIT binary patch literal 460 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!63?wyl`GbKJOS+@4BLl;=8&~`zjDUQ}64!{5 z;QX|b^2DN42FH~Aq*MjZ+{E<Mpwz^a%EFVWHb6yD0X`wFE+MINE`9S#t2456o^<}3 zj-i#Dir(tW_tssyv-H%p2QS{7I(h2a%eT+pe}4bz%ad1cEnR}A?LYhN=dY$Udly~( zcH;T(jkkW}PF%M8!OyMteyqCr{ovzY|Ns978vk!K_g5e#R}$nG3>3i)7~V+8!~s>F z^K@|xskoK&Ao2MF5fKqKrozHP$H>SAhNOh)j!aFDHZd?BIiRS_#db+SLGK)kfCJAQ zqeTU>kA$3ffHDdi%#){0m?9#a#K57jfnf&sBX3Sa1{Q(l4Luqi?2o!UJvli+Tm$)4 z3=Ry8tOs@-j1d%!V{BkJBmmUGR=9zogZqi`CnIJS0S|@~)0;j>{+!9k)ZrMzp`^g{ zWb=lCjZ7>85e%#3G!7i+5^!)}RE&#@t221;=y4h|P`{c&LV^GTQ(9VBm>F9G12cn9 XiE#1KH6NpaK4b87^>bP0l+XkKLQcLN literal 0 HcmV?d00001 diff --git a/plugins/content_types/term_context/term_description.inc b/plugins/content_types/term_context/term_description.inc new file mode 100644 index 00000000..7a2311ac --- /dev/null +++ b/plugins/content_types/term_context/term_description.inc @@ -0,0 +1,51 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function ctools_term_description_ctools_content_types() { + return array( + 'single' => TRUE, + 'title' => t('Term description'), + 'icon' => 'icon_term.png', + 'description' => t('Term description.'), + 'required context' => new ctools_context_required(t('Term'), 'term'), + 'category' => t('Taxonomy term'), + ); +} + +function ctools_term_description_content_type_render($subtype, $conf, $panel_args, $context) { + $term = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'node_type'; + + $block->title = $term->name; + if ($term) { + $block->content = _filter_autop(filter_xss_admin($term->description)); + $block->delta = $term->tid; + + if (user_access('administer taxonomy')) { + $block->admin_links['update'] = array( + 'title' => t('Edit term'), + 'alt' => t("Edit this term"), + 'href' => "admin/content/taxonomy/edit/term/$term->tid", + 'query' => drupal_get_destination(), + ); + } + } + else { + $block->content = t('Term description goes here.'); + $block->delta = 'unknown'; + } + + return $block; +} + +function ctools_term_description_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" term description', array('@s' => $context->identifier)); +} + +function ctools_term_description_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/term_context/term_list.inc b/plugins/content_types/term_context/term_list.inc new file mode 100644 index 00000000..f45ebaaa --- /dev/null +++ b/plugins/content_types/term_context/term_list.inc @@ -0,0 +1,114 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function panels_term_list_panels_content_types() { + return array( + 'single' => TRUE, + 'title' => t('List of related terms'), + 'icon' => 'icon_term.png', + 'description' => t('Terms related to an existing term; may be child, siblings or top level.'), + 'required context' => new ctools_context_required(t('Term'), 'term'), + 'category' => t('Taxonomy term'), + 'defaults' => array('title' => '', 'type' => 'child', 'list_type' => 'ul'), + ); + return $items; +} + +function ctools_term_list_content_type_render($subtype, $conf, $panel_args, $context) { + $term = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'term-list'; + + $options = panels_admin_term_list_options(); + if ($term) { + $block->subject = $options[$conf['type']]; + $block->delta = $conf['type']; + switch ($conf['type']) { + case 'related': + $terms = taxonomy_get_related($term->tid); + break; + + case 'child': + default: + $terms = taxonomy_get_children($term->tid); + break; + + case 'top': + $terms = taxonomy_get_children(0, $term->vid); + break; + + case 'sibling': + $parent = db_result(db_query("SELECT parent FROM {term_hierarchy} WHERE tid = %d", $term->tid)); + $terms = taxonomy_get_children($parent, $term->vid); + // Remove the term that started this. + unset($terms[$term->tid]); + break; + + case 'synonyms': + $terms = taxonomy_get_synonyms($term->tid); + break; + } + if ($terms) { + foreach ($terms as $related) { + $items[$related->tid] = l($related->name, taxonomy_term_path($related), array('rel' => 'tag', 'title' => strip_tags($related->description))); + } + + $block->content = theme('item_list', $items, NULL, $conf['list_type']); + } + } + else { + $block->content = t('Term description goes here.'); + $block->delta = 'unknown'; + } + + return $block; +} + +function panels_admin_term_list_options() { + return array( + 'child' => t('Child terms'), + 'related' => t('Related terms'), + 'sibling' => t('Sibling terms'), + 'top' => t('Top level terms'), + 'synonyms' => t('Term synonyms'), + ); +} + +/** + * Returns an edit form for the custom type. + */ +function ctools_term_list_content_type_edit_form(&$form, &$form_state) { + $conf = $form_state['conf']; + + $form['type'] = array( + '#type' => 'radios', + '#title' => t('Which terms'), + '#options' => panels_admin_term_list_options(), + '#default_value' => $conf['type'], + '#prefix' => '<div class="clear-block no-float">', + '#suffix' => '</div>', + ); + + $form['list_type'] = array( + '#type' => 'select', + '#title' => t('List type'), + '#options' => array('ul' => t('Unordered'), 'ol' => t('Ordered')), + '#default_value' => $conf['list_type'], + ); +} + +function ctools_term_list_content_type_admin_title($subtype, $conf, $context) { + $options = panels_admin_term_list_options(); + return t('"@s" @type', array('@s' => $context->identifier, '@type' => drupal_strtolower($options[$conf['type']]))); +} + +function ctools_term_list_content_type_edit_form_submit(&$form, &$form_state) { + // Copy everything from our defaults. + foreach (array_keys($form_state['plugin']['defaults']) as $key) { + $form_state['conf'][$key] = $form_state['values'][$key]; + } +} + diff --git a/plugins/content_types/user_context/icon_user.png b/plugins/content_types/user_context/icon_user.png new file mode 100644 index 0000000000000000000000000000000000000000..ab248f3f1bb62536bbf23296a949bfe05cd56bc7 GIT binary patch literal 606 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!63?wyl`GbKJOS+@4BLl;=8&~`zjDUQ}64!{5 z;QX|b^2DN42FH~Aq*MjZ+{E<Mpwz^a%EFVWHb6y-0(?STjVzs~?LWKw!OuHyzrX+V z<@x*1-+unO@%p=tq1ET_KUZJAxAfGtjkkWRyL9L3o8NOTecO8P$D(qtb1#2KXI5N% z^*g_zuPi4yG06GJtGDNtHfNOAzkhV?<kh<;pa1^!;@-*GrKfK^D4)0S!<TP2-u}7! z;m@b9-&{gcEy9`)Ps#8~t6OyS+llAD4?h0YnP{}J%yrex?@ep==1yGp^7=7Lm*D^Z z{{x+QzcV)+NST!c`GG`82Mi1Q{R|lx7#TcW978H@B^f+@^fWESgn`Lf!N8Q2LxD|C zgwZ#+fg$OE=|wiCScMB0FB%$NWC2R#38yK<$NBsFPw(vP^yO7(kdSC(+BK<4VS)C6 z!^_wN90V9`l@%^retI!Im>H;^SvXDBF|$xvxlq}$w$>~eXr@6z;<bwtgpG}aj4v^8 zGO!2)G{m}lF)wj-j`i-oDxuJDfXU~Tl*58`{6Y*s6PzbBG*}5R{QR?qhk=nrC_(7K zMRq2I`SDGRK+Ac2O-!#|>1k|eY;1PE2-GXVV;X5`X&HLZ!861S#P>0c44v}x#}5~m smX;J((H2Igj)PZ3BD*&lIy5jc$ag6`Uyzy~2n<aIPgg&ebxsLQ0Lelg8~^|S literal 0 HcmV?d00001 diff --git a/plugins/content_types/user_context/profile_fields.inc b/plugins/content_types/user_context/profile_fields.inc new file mode 100644 index 00000000..31facf8b --- /dev/null +++ b/plugins/content_types/user_context/profile_fields.inc @@ -0,0 +1,132 @@ +<?php +// $Id$ + + +/** + * Callback function to supply a list of content types. + */ +function panels_profile_fields_panels_content_types() { + if (module_exists('profile') && !is_null(profile_categories())) { + $items['profile_fields'] = array( + 'single' => TRUE, + 'title' => t('Profile category'), + 'icon' => 'icon_user.png', + 'description' => t('Contents of a single profile category.'), + 'required context' => new ctools_context_required(t('User'), 'user'), + 'category' => t('User'), + 'defaults' => array('title' => '', 'category' => '', 'empty' => ''), + 'hook theme' => 'ctools_profile_fields_content_type_theme', + ); + return $items; + } +} + +/** + * 'Render' callback for the 'profile fields' content type. + */ +function ctools_profile_fields_content_type_render($subtype, $conf, $panel_args, $context) { + $account = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'profile fields'; + + if ($account) { + // Get the category from the options + $category = str_replace("_", " ", $conf['category']); + + // Set the subject to the name of the category + $block->subject = $category; + + // Put all the fields in the category into an array + profile_view_profile($account); + + if (is_array($account->content[$category])) { + foreach ($account->content[$category] as $field) { + if (is_array($field['#attributes'])) { + $vars[$field['#attributes']['class']]['title'] = $field['#title']; + $vars[$field['#attributes']['class']]['value'] = $field['#value']; + } + } + } + + if (count($vars) == 0) { + // Output the given empty text + $output = $conf['empty']; + } + else { + // Call the theme function with the field vars + $output = theme('profile_fields_pane', $category, $vars); + } + + $block->content = $output; + $block->delta = $account->uid; + } + else { + $block->subject = $conf['category']; + $block->content = t('Profile content goes here.'); + $block->delta = 'unknown'; + } + + return $block; +} +/** + * Helper function : build the list of categories for the 'edit' form. + */ +function _ctools_profile_fields_options() { + $cat_list = array(); + + $categories = profile_categories(); + foreach ($categories as $key => $value) { + $cat_list[str_replace(" ", "_", $value['name'])] = $value['title']; + } + + return $cat_list; +} + +/** + * 'Edit' callback for the 'profile fields' content type. + */ +function ctools_profile_fields_content_type_edit_form(&$form, &$form_state) { + $conf = $form_state['conf']; + $form['category'] = array( + '#type' => 'radios', + '#title' => t('Which category'), + '#options' => _ctools_profile_fields_options(), + '#default_value' => $conf['category'], + '#prefix' => '<div class="clear-block no-float">', + '#suffix' => '</div>', + ); + + $form['empty'] = array( + '#type' => 'textarea', + '#title' => 'Empty text', + '#description' => t('Text to display if category has no data. Note that title will not display unless overridden.'), + '#rows' => 5, + '#default_value' => $conf['empty'], + '#prefix' => '<div class="clear-block no-float">', + '#suffix' => '</div>', + ); + + return $form; +} + +function ctools_profile_fields_content_type_edit_form_submit(&$form, &$form_state) { + // Copy everything from our defaults. + foreach (array_keys($form_state['plugin']['defaults']) as $key) { + $form_state['conf'][$key] = $form_state['values'][$key]; + } +} + +/** + * 'Title' callback for the 'profile fields' content type. + */ +function ctools_profile_fields_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" profile fields', array('@s' => $conf['category'])); +} + +function ctools_profile_fields_content_type_theme(&$theme, $plugin) { + $theme['profile_fields_pane'] = array( + 'arguments' => array('category' => NULL, 'vars' => NULL), + 'path' => $plugin['path'], + 'template' => 'profile_fields_pane', + ); +} diff --git a/plugins/content_types/user_context/profile_fields_pane.tpl.php b/plugins/content_types/user_context/profile_fields_pane.tpl.php new file mode 100644 index 00000000..f49866bc --- /dev/null +++ b/plugins/content_types/user_context/profile_fields_pane.tpl.php @@ -0,0 +1,17 @@ +<?php +// $Id$ +/** + * @file + * Display profile fields. + * + * @todo Need definition of what variables are available here. + */ +?> +<?php if (is_array($vars)): ?> + <?php foreach ($vars as $class => $field): ?> + <dl class="profile-category"> + <dt class="profile-<?php print $class; ?>"><?php print $field['title']; ?></dt> + <dd class="profile-<?php print $class; ?>"><?php print $field['value']; ?></dd> + </dl> + <?php endforeach; ?> +<?php endif; ?> diff --git a/plugins/content_types/user_context/user_picture.inc b/plugins/content_types/user_context/user_picture.inc new file mode 100644 index 00000000..e5a0d4bd --- /dev/null +++ b/plugins/content_types/user_context/user_picture.inc @@ -0,0 +1,43 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function ctools_user_picture_ctools_content_types() { + $items['user_picture'] = array( + 'single' => TRUE, + 'title' => t('User picture'), + 'icon' => 'icon_user.png', + 'description' => t('The picture of a user.'), + 'required context' => new ctools_context_required(t('User'), 'user'), + 'category' => t('User'), + ); + return $items; +} + +function ctools_user_picture_content_type_render($subtype, $conf, $panel_args, $context) { + $account = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'term-list'; + + if ($account === FALSE || ($account->access == 0 && !user_access('administer users'))) { + return drupal_not_found(); + } + + $block->title = check_plain($account->name); + $block->content = theme('user_picture', $account); + + return $block; +} + +/** + * Display the administrative title for a panel pane in the drag & drop UI + */ +function ctools_user_picture_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" user picture', array('@s' => $context->identifier)); +} + +function ctools_user_picture_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} diff --git a/plugins/content_types/user_context/user_profile.inc b/plugins/content_types/user_context/user_profile.inc new file mode 100644 index 00000000..c15c3a8f --- /dev/null +++ b/plugins/content_types/user_context/user_profile.inc @@ -0,0 +1,59 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function ctools_user_profile_ctools_content_types() { + return array( + 'single' => TRUE, + 'title' => t('User profile'), + 'icon' => 'icon_user.png', + 'description' => t('The profile of a user.'), + 'required context' => new ctools_context_required(t('User'), 'user'), + 'category' => t('User'), + ); + return $items; +} + +function ctools_user_profile_content_type_render($subtype, $conf, $panel_args, $context) { + $account = isset($context->data) ? drupal_clone($context->data) : NULL; + $block = new stdClass(); + $block->module = 'user-profile'; + + if ($account === FALSE || ($account->access == 0 && !user_access('administer users'))) { + return drupal_not_found(); + } + // Retrieve and merge all profile fields: + $fields = array(); + foreach (module_list() as $module) { + if ($data = module_invoke($module, 'user', 'view', '', $account)) { + foreach ($data as $category => $items) { + foreach ($items as $key => $item) { + $item['class'] = "$module-". $item['class']; + $fields[$category][$key] = $item; + } + } + } + } + + drupal_alter('profile', $account); + + $block->title = check_plain($account->name); + $block->content = theme('user_profile', $account, $fields); + + return $block; +} + + +/** + * Display the administrative title for a panel pane in the drag & drop UI + */ +function ctools_user_profile_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" user profile', array('@s' => $context->identifier)); +} + +function ctools_user_profile_content_type_edit_form(&$form, &$form_state) { + // provide a blank form so we have a place to have context setting. +} + diff --git a/plugins/content_types/vocabulary_context/icon_vocabulary.png b/plugins/content_types/vocabulary_context/icon_vocabulary.png new file mode 100644 index 0000000000000000000000000000000000000000..f0417cb6515aa7fb2856c90722b386ed99bfc501 GIT binary patch literal 460 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!63?wyl`GbKJOS+@4BLl;=8&~`zjDUQ}64!{5 z;QX|b^2DN42FH~Aq*MjZ+{E<Mpwz^a%EFVWHb6yD0X`wFE+MINE`9S#t2456o^<}3 zj-i#Dir(tW_tssyv-H%p2QS{7I(h2a%eT+pe}4bz%ad1cEnR}A?LYhN=dY$Udly~( zcH;T(jkkW}PF%M8!OyMteyqCr{ovzY|Ns978vk!K_g5e#R}$nG3>3i)7~V+8!~s>F z^K@|xskoK&Ao2MF5fKqKrozHP$H>SAhNOh)j!aFDHZd?BIiRS_#db+SLGK)kfCJAQ zqeTU>kA$3ffHDdi%#){0m?9#a#K57jfnf&sBX3Sa1{Q(l4Luqi?2o!UJvli+Tm$)4 z3=Ry8tOs@-j1d%!V{BkJBmmUGR=9zogZqi`CnIJS0S|@~)0;j>{+!9k)ZrMzp`^g{ zWb=lCjZ7>85e%#3G!7i+5^!)}RE&#@t221;=y4h|P`{c&LV^GTQ(9VBm>F9G12cn9 XiE#1KH6NpaK4b87^>bP0l+XkKLQcLN literal 0 HcmV?d00001 diff --git a/plugins/content_types/vocabulary_context/vocabulary_terms.inc b/plugins/content_types/vocabulary_context/vocabulary_terms.inc new file mode 100644 index 00000000..13b152a0 --- /dev/null +++ b/plugins/content_types/vocabulary_context/vocabulary_terms.inc @@ -0,0 +1,96 @@ +<?php +// $Id$ + +/** + * Callback function to supply a list of content types. + */ +function ctools_vocabulary_terms_ctools_content_types() { + if (module_exists('taxonomy')) { + return array( + 'single' => TRUE, + 'title' => t('Vocabulary terms'), + 'icon' => 'icon_vocabulary.png', + 'description' => t('All the terms in a vocabulary.'), + 'required context' => new ctools_context_required(t('Vocabulary'), 'vocabulary'), + 'category' => t('Vocabulary'), + 'defaults' => array('max_depth' => 0, 'tree' => 1), + ); + return $items; + } +} + +/** + * Output function for the 'vocabulary terms' content type. Outputs a + * list of terms for the input vocabulary. + */ +function ctools_vocabulary_terms_content_type_render($subtype, $conf, $panel_args, $context) { + $vocab = isset($context->data) ? drupal_clone($context->data) : NULL; + $max_depth = (!empty($conf['max_depth']) ? (int)$conf['max_depth'] : NULL); + if ($conf['tree'] == FALSE) { + $terms = taxonomy_get_tree($vocab->vid, 0, -1, $max_depth); + $items = array(); + foreach ($terms as $term) { + $items[] = l($term->name, 'taxonomy/term/' . $term->tid); + } + $output = theme('item_list', $items); + } + else { + $output = theme('item_list', _ctools_content_vocabulary_terms($vocab->vid, $max_depth)); + } + + $block = new stdClass(); + $block->module = 'node_type'; + $block->title = check_plain($vocab->name); + $block->content = $output; + $block->delta = $vocab->tid; + + return $block; +} + +function _ctools_content_vocabulary_terms($vid, $max_depth, $depth = -1, $tid = 0) { + $depth++; + if ($max_depth != NULL && $depth == $max_depth) { + return array(); + } + $return = array(); + $query = db_query('SELECT t.name, t.tid FROM {term_data} t INNER JOIN {term_hierarchy} h ON t.tid = h.tid WHERE t.vid = %d AND h.parent = %d ORDER BY t.weight ASC, t.name ASC', $vid, $tid); + while ($result = db_fetch_object($query)) { + $return[] = array( + 'data' => l($result->name, 'taxonomy/term/'. $result->tid), + 'children' => _ctools_content_vocabulary_terms($vid, $max_depth, $depth, $result->tid), + ); + } + return $return; +} + +function ctools_vocabulary_terms_content_type_admin_title($subtype, $conf, $context) { + return t('"@s" terms', array('@s' => $context->identifier)); +} + +function ctools_vocabulary_terms_content_type_edit_form(&$form, &$form_state) { + $conf = $form_state['conf']; + $form['max_depth'] = array( + '#type' => 'select', + '#title' => t('Maximum depth'), + '#options' => array_merge(array(t('unlimited')), range(1, 9)), + '#default_value' => $conf['max_depth'], + '#description' => t('Define the maximum depth of terms being displayed.'), + ); + + $form['tree'] = array( + '#type' => 'checkbox', + '#title' => t('Display as tree'), + '#default_value' => $conf['tree'], + '#description' => t('If checked, the terms are displayed in a tree, otherwise in a flat list.'), + ); + + return $form; +} + +function ctools_vocabulary_terms_content_type_edit_form_submit(&$form, &$form_state) { + // Copy everything from our defaults. + foreach (array_keys($form_state['plugin']['defaults']) as $key) { + $form_state['conf'][$key] = $form_state['values'][$key]; + } +} + -- GitLab