'custom_breadcrumb', 'field' => 'node_type', 'type' => 'node', 'name_constructor' => '_custom_breadcrumbs_breadcrumb_name', ); return $breadcrumb_type_info; } /** * Constructs a default name to display in the admin screen. */ function _custom_breadcrumbs_breadcrumb_name($breadcrumb) { if (isset($breadcrumb->node_type)) { return $breadcrumb->node_type; } } /** * Implements hook_theme(). */ function custom_breadcrumbs_theme() { return array( 'custom_breadcrumbs_filter_form' => array( 'arguments' => array('form' => array()), 'file' => 'custom_breadcrumbs.admin.inc', ), 'custom_breadcrumbs_filters' => array( 'arguments' => array('form' => array()), 'file' => 'custom_breadcrumbs.admin.inc', ), 'custom_breadcrumbs_help_identifiers' => array( 'arguments' => array(), ), 'custom_breadcrumbs_module_weight' => array( 'arguments' => array('form' => array()), 'file' => 'custom_breadcrumbs.admin.inc', ), ); } /** * Implements hook_menu(). */ function custom_breadcrumbs_menu() { $items = array(); $items['admin/build/custom_breadcrumbs'] = array( 'title' => 'Custom Breadcrumbs', 'description' => 'Custom breadcrumbs', 'page callback' => 'custom_breadcrumbs_page', 'access arguments' => array('administer custom breadcrumbs'), 'file' => 'custom_breadcrumbs.admin.inc', ); $items['admin/build/custom_breadcrumbs/list'] = array( 'title' => 'List', 'page callback' => 'custom_breadcrumbs_page', 'access arguments' => array('administer custom breadcrumbs'), 'file' => 'custom_breadcrumbs.admin.inc', 'type' => MENU_DEFAULT_LOCAL_TASK, 'weight' => 0, ); $items['admin/build/custom_breadcrumbs/node/add'] = array( 'title' => 'Node', 'page callback' => 'drupal_get_form', 'page arguments' => array('custom_breadcrumbs_form', 'node'), 'access arguments' => array('administer custom breadcrumbs'), 'file' => 'custom_breadcrumbs.admin.inc', 'type' => MENU_LOCAL_TASK, 'weight' => 1, ); $items['admin/build/custom_breadcrumbs/node/edit'] = array( 'title' => 'Edit custom breadcrumb for nodes', 'page callback' => 'drupal_get_form', 'page arguments' => array('custom_breadcrumbs_form', 'node'), 'access arguments' => array('administer custom breadcrumbs'), 'file' => 'custom_breadcrumbs.admin.inc', 'type' => MENU_CALLBACK, ); $items['admin/settings/custom-breadcrumbs'] = array( 'title' => 'Custom Breadcrumb Settings', 'page callback' => 'drupal_get_form', 'page arguments' => array('custom_breadcrumbs_admin_settings'), 'access arguments' => array('administer custom breadcrumbs'), 'file' => 'custom_breadcrumbs.admin.inc', 'type' => MENU_NORMAL_ITEM, ); return $items; } /** * Implements hook_preprocess(). */ function custom_breadcrumbs_preprocess(&$variables, $hook) { static $tried = array(); // Only respond to the first call for this hook. if (!isset($tried[$hook])) { $tried[$hook] = TRUE; if (!custom_breadcrumbs_exclude_path()) { if (variable_get('custom_breadcrumbs_set_global_home_breadcrumb', FALSE)) { $trail = drupal_get_breadcrumb(); if (!empty($trail)) { array_shift($trail); $cb_home = custom_breadcrumbs_home_crumb(); if (!empty($cb_home)) { array_unshift($trail, array_pop($cb_home)); } drupal_set_breadcrumb($trail); $variables['breadcrumb'] = theme('breadcrumb', drupal_get_breadcrumb()); } } if (variable_get('custom_breadcrumbs_set_menu_breadcrumb', FALSE)) { // Use default menu structure to set the breadcrumb. if (custom_breadcrumbs_set_menu_breadcrumb()) { $variables['breadcrumb'] = theme('breadcrumb', drupal_get_breadcrumb());; } } } } } /** * Implements hook_perm(). */ function custom_breadcrumbs_perm() { return array('administer custom breadcrumbs', 'use php in custom breadcrumbs'); } /** * Implements hook_help(). */ function custom_breadcrumbs_help($path, $arg) { switch ($path) { case 'admin/help#custom_breadcrumbs': $output = '

'. t("Custom Breadcrumbs allows you to create and modify your own breadcrumbs based on node type. After enabling the module, click on Administer > Site building > Custom breadcrumbs. Select Node Type at the top of the page to create breadcrumbs according to node type. Other types (such as views, paths, and taxonomy) may also be present, depending on what modules have been installed.") .'

'; $output .= '

'. t("On the Node Type page, select the node type the breadcrumb will apply to. There are two text fields below-- 'Titles' and 'Paths.' When creating a breadcrumb, you are simply creating a link. In the custom breadcrumbs interface 'Titles' describes the text of the breadcrumb while 'Paths' describes the Drupal path the breadcrumb links to. Each Title must have a corresponding Path.") .'

'; $output .= '

'. t("To give a very simple example of how to use this module, let's say I have a blog on my web site called 'Deep Thoughts.' To create this, I use the Views module to create a page at /blog that displays all the node types 'blog post.' Whenever a user views a blog post I want the breadcrumb to show Home > Deep Thoughts instead of simply Home. To do this I would simply type 'Deep Thoughts' in the 'Titles' field and and 'blog' in the 'Paths' field and save my breadcrumb.") .'

'; $output .= '

'. t("Using the Tokens module, the Custom breadcrumbs module becomes much more flexible because breadcrumbs can become dynamic. You can create a breadcrumb like Home > Deep Thoughts > [Month of Blog Post] [Year of Blog Post], where 'Deep Thoughts' links to my main blog page and '[Month of Blog Post] [Year of Blog Post]' links to a view that shows only blog posts from the month and year the blog post was created (e.g. June 2007). For this, you would do the following:") .'

'; $output .= '

'. t("Node Type:
Blog Post

Titles:
Deep Thoughts
[month] [yyyy]

Paths:
blog
blog/[mm]_[yyyy]
(where of course, blog/[mm]_[yyyy] is the path to the view of blog posts from that month and year). So if you created a blog post on June 13, 2007 your breadcrumb would show Home > Deep Thoughts > June 2007 and 'June 2007' links to 'blog/06_2007' which is a view of all blog posts from June 2007.") .'

'; $output .= '

'. t("Also, note that Custom Breadcrumbs doesn't actually check to be sure that a particular path exists, so you'll have to check yourself to avoid 404 errors.") .'

'; $output .= '

'. t("Only users with 'administer custom breadcrumbs' permission will be allowed to create or modify custom breadcrumbs.") .'

'; $output .= '

'. t("Breadcrumb Visibility") .'

'; $output .= '

'. t("Users given 'use php in custom breadcrumbs' permission can include php code snippet that returns TRUE or FALSE to control whether or not the breadcrumb is displayed. Note that this code has access to the %node variable, and can check its type or any other property.", array('%node' => '$node')) .'

'; $output .= '

'. t("Special Identifiers") .'

'; $output .= '

'. t("The following identifiers can be used to achieve a special behavior:"); $output .= '

'; $output .= '

'. t("Identifiers should be added to the paths area in the following format: identifier|path. To be recognized, the identifier must be enclosed in angular brackets and proceed any part of the path. For example: %pathauto|[ogname-raw]", array('%pathauto' => '')) .'

'; return $output; case 'admin/build/custom_breadcrumbs': $output = '

'. t("To create a custom breadcrumb, choose one of the breadcrumb types listed above. The following table lists all of the custom breadcrumbs that have been defined. The list can be filtered by breadcrumb type or language, or sorted by clicking on one of the column headings.") .'

'; $output .= '

'. t('You can configure Custom Breadcrumb settings at admin/settings/custom-breadcrumbs.', array('@link' => url('admin/settings/custom-breadcrumbs'))) .'

'; return $output; } } /** * Implements hook_nodeapi(). */ function custom_breadcrumbs_nodeapi($node, $op, $teaser, $page) { if ($op == 'alter' && empty($teaser) && !empty($page)) { // Check for breadcrumb for this node type. global $language; $languages = array('language' => $language->language, 'all' => ''); $breadcrumbs = custom_breadcrumbs_load_breadcrumbs('custom_breadcrumbs', NULL, array('node_type' => $node->type), $languages); if (!empty($breadcrumbs)) { $objs = array('node' => $node); if ($breadcrumb = custom_breadcrumbs_select_breadcrumb($breadcrumbs, $objs)) { custom_breadcrumbs_set_breadcrumb($breadcrumb, $objs); } } } } /** * Implements hook_form_alter(). */ function custom_breadcrumbs_form_alter(&$form, $form_state, $form_id) { // Provide custom breadcrumbs for comment forms associated with nodes. if ($form_id == 'comment_form' && isset($form['nid']['#value'])) { $node = node_load($form['nid']['#value']); // Call custom_breadcrumbs_nodeapi to provide a custom_breadcrumb for this comment. custom_breadcrumbs_nodeapi($node, 'alter', array(), array(1)); } elseif (user_access('administer custom breadcrumbs') && isset($form['type']) && isset($form['#node']) && $form['type']['#value'] .'_node_form' == $form_id) { // Provide a custom breadcrumbs fieldset for node edit forms. $node = $form['#node']; // Load all custom breadcrumbs for this node type. $breadcrumbs = custom_breadcrumbs_load_breadcrumbs('custom_breadcrumbs', 'custom_breadcrumb', array('node_type' => $form['type']['#value'])); foreach (module_implements('cb_node_form_table') as $module) { $func = $module .'_cb_node_form_table'; $more = $func($node); if (!empty($more)) { $breadcrumbs = array_merge($breadcrumbs, $more); } } $output = NULL; if (count($breadcrumbs) > 0) { $output = '

'. t('Custom breadcrumbs have been created for this %type page. Use the Custom Breadcrumbs Administration Page to add additional breadcrumbs, or follow the links in the table below to edit or delete existing custom breadcrumbs.', array('%type' => $form['type']['#value'], '@link' => url('admin/build/custom_breadcrumbs'))) .'

'; } // Show a table of custom breadcrumbs with links to the edit form. module_load_include('inc', 'custom_breadcrumbs', 'custom_breadcrumbs.admin'); $output .= custom_breadcrumbs_simple_breadcrumb_table($breadcrumbs); $form['custom_breadcrumbs'] = array( '#type' => 'fieldset', '#title' => t('Custom Breadcrumbs'), '#access' => user_access('administer custom breadcrumbs'), '#group' => 'additional_settings', '#collapsible' => TRUE, '#collapsed' => TRUE, '#weight' => -50, ); $form['custom_breadcrumbs']['breadcrumb_table'] = array('#value' => $output, ); } } /** * Selects a breadcrumb from an array of breadcrumbs. * * @param $breadcrumbs * The array of breadcrumb objects that the breadcrumb will be selected from. * @param $objs * An array of optional object (node, view, ...) to aid in the selection process. * * @return $breadcrumb * The selected breadcrumb object. */ function custom_breadcrumbs_select_breadcrumb($breadcrumbs, $objs = array()) { while ($breadcrumb = array_pop($breadcrumbs)) { if (custom_breadcrumbs_is_visible($breadcrumb, $objs)) { return $breadcrumb; } } } /** * Sets the custom breadcrumb. * * This can be used by submodules, but they could also provide their own function. * @param $breadcrumb * The breadcrumb object * @param $objs * An array of objects (if available) for building token substituions. */ function custom_breadcrumbs_set_breadcrumb($breadcrumb, $objs = array()) { if ($breadcrumb && !custom_breadcrumbs_exclude_path()) { $locations = array(); $trail = _custom_breadcrumbs_get_breadcrumb($breadcrumb, $objs, $locations); if (variable_get('custom_breadcrumbs_force_active_trail', FALSE)) { menu_set_active_trail($locations); } drupal_set_breadcrumb($trail); // Optionally save the unique breadcrumb id of the last set breadcrumb. custom_breadcrumbs_unique_breadcrumb_id($breadcrumb->breadcrumb_type, $breadcrumb->bid); } } /** * Gets the custom breadcrumb. * * This function is used to retrieve the breadcrumb trail before actually setting it. * @see custom_breadcrumbs_get_breadcrumb(). * * @param $breadcrumb * The breadcrumb object. * @param $objs * An array of object (if available) for building token substituions. * @param $locations * Locations array to be able to set active menu trail - passed by reference. * * @return * array of html crumbs */ function _custom_breadcrumbs_get_breadcrumb($breadcrumb, $objs, &$locations) { // Assure locations is an array. if (!is_array($locations)) { $locations = array(); } // Avoid invalid nesting. elseif (isset($locations['title']) || isset($locations['href'])) { $locations = array($locations); } $trail = custom_breadcrumbs_home_crumb(); if (!empty($trail)) { $title = variable_get('custom_breadcrumb_home', t('Home')); $locations[] = array('title' => variable_get('custom_breadcrumb_home', t('Home')), 'href' => '', 'localized_options' => array()); } if (variable_get('custom_breadcrumbs_use_php_in_titles', FALSE)) { $titles = extract_php($breadcrumb->titles, $objs); // Titles and paths arrays can also be provided as elements of an associative array. if (isset($titles['titles']) && is_array($titles['titles']) && isset($titles['paths']) && is_array($titles['paths'])) { $paths = $titles['paths']; $titles = $titles['titles']; } else { $paths = extract_php($breadcrumb->paths, $objs); } } if (!isset($titles) || is_null($titles)) { $titles = preg_split("/[\n]+/", $breadcrumb->titles); } if (!isset($paths) || is_null($paths)) { $paths = preg_split("/[\n]+/", $breadcrumb->paths); } // Token replacement for titles and paths if (module_exists('token')) { // Do token replacement. $types = custom_breadcrumbs_token_types($objs); $titles = token_replace_multiple($titles, $types); $paths = token_replace_multiple($paths, $types); } $items = _custom_breadcrumbs_get_trail_items($breadcrumb, $titles, $paths); // Use the returned items to set the trail. foreach ($items as $item) { if ($item['crumb']) { $trail[] = $item['crumb']; } if ($set_active_trail) { // TODO: only do this if $item['href'] is set? $locations[] = array('title' => $item['title'], 'href' => drupal_get_normal_path(trim($item['href']))); } } return $trail; } /** * Builds the trail items for a given breadcrumb specification. * * @param $breadcrumb * The breadcrumb object. * @param $titles * An array of titles (after token replacement). * @param $paths * An array of paths (after token replacement) that may contain special identifiers. * * @return * An associative array of trail items with keys * 'title' - the title of the item * 'href' - the path of the item used to set the active trail * 'crumb'- the html crumb for use in the breadcrumb */ function _custom_breadcrumbs_get_trail_items($breadcrumb, $titles, $paths) { $trail_items = array(); for ($i = 0; $i < count($titles); $i++) { $title = trim($titles[$i]); if (($title != '') && ($title != '')) { // Create a breadcrumb only if there is a title. // Include optional html attributes. $options = _custom_breadcrumbs_identifiers_option($i+1, $breadcrumb->bid); $crumb_items = _custom_breadcrumbs_create_crumb_items($title, trim($paths[$i]), $options); $trail_items = array_merge($trail_items, $crumb_items); } } return $trail_items; } /** * Sets or gets the unique breadcrumb id. * * @param $type * The breadcrumb type, used to set the unique breadcrumb id. * @param $bid * The breadcrumb id, used to set the unique breadcrumb id. * * @return $cbid * A string containing the unique id for this breadcrumb. */ function custom_breadcrumbs_unique_breadcrumb_id($type = NULL, $bid = NULL) { static $stored_breadcrumb_id; if (variable_get('custom_breadcrumbs_type_class', FALSE)) { if (isset($type)) { $base = 'custom-breadcrumbs'; $cbid = $base .'-'. $type; if (variable_get('custom_breadcrumbs_append_bid_class', FALSE) && isset($bid)) { $cbid .= '-'. $bid; } $stored_breadcrumb_id = $cbid; } else { $cbid = $stored_breadcrumb_id; } return $cbid; } } /** * Prepares some common contexts for token substitution. * * @param $objs * An array of objects to be used in token replacement. Array keys indicate type of object. * * @return $types * An array of substitution classes for token_replace_multiple(). */ function custom_breadcrumbs_token_types($objs = array()) { if (!isset($objs['user'])) { global $user; if ($user->uid) { $user = user_load(array('uid' => $user->uid)); } $objs['user'] = $user; } $objs['global'] = NULL; return $objs; } /** * Saves the custom breadcrumb. * * @param $module * The name of the custom breadcrumbs submodule that created the breadcrumb. * @param $key * The type of breadcrumb to save. * @param $breadcrumb * Any additional submodule function to call after breadcrumb has been saved. */ function _custom_breadcrumbs_save_breadcrumb($module, $key, $breadcrumb) { if (is_array($breadcrumb->paths)) { $breadcrumb->paths = implode("\n", $breadcrumb->paths); } if (is_array($breadcrumb->titles)) { $breadcrumb->titles = implode("\n", $breadcrumb->titles); } $info = module_invoke($module, 'cb_breadcrumb_info'); if (isset($info[$key])) { if ((!isset($breadcrumb->name) || $breadcrumb->name == '') && isset($info[$key]['name_constructor']) && function_exists($info[$key]['name_constructor'])) { $breadcrumb->name = $info[$key]['name_constructor']($breadcrumb); } if (isset($breadcrumb->bid)) { drupal_write_record($info[$key]['table'], $breadcrumb, 'bid'); } else { drupal_write_record($info[$key]['table'], $breadcrumb); } } } /** * Deletes the custom breadcrumb. * * @param $module * The name of the custom breadcrumbs submodule that created the breadcrumb. * @param $key * An array key indicating the type of custom breadrumb that is to be deleted. * @param $bid * The id for the breadcrumb that is to be deleted. */ function _custom_breadcrumbs_delete_breadcrumb($module, $key, $bid) { $info = module_invoke($module, 'cb_breadcrumb_info'); if (isset($info[$key]['table'])) { db_query('DELETE FROM {'. $info[$key]['table'] .'} WHERE bid = %d', $bid); } } /** * Create the Home breadcrumb trail. * * @return * The home breadcrumb item. */ function custom_breadcrumbs_home_crumb() { $hometext = variable_get('custom_breadcrumb_home', t('Home')); if ($hometext != '') { // Add any html identifiers. $options = _custom_breadcrumbs_identifiers_option(); $trail = array(l($hometext, '', $options)); } else { $trail = array(); } return $trail; } /** * Creates one or more crumb items out of a custom breadcrumb definition line. * * @param $title * Title string of the custom breadcrumb definition (after token replacement). * @param $original_path * Path string of the custom breadcrumb definition (after token replacment) * which may contain a special identifier. * @param $attributes * An array of additional attributes for the breadcrumb item. * @return * An array of one or multiple crumb items. * In most cases, especially without an identifier, it is only an array of one item. */ function _custom_breadcrumbs_create_crumb_items($title, $original_path, $attributes = array()) { // The array to return. $crumbs = array(); // Decode title to properly handle special characters. $title = decode_entities($title); // Collapse double slashes to one. $original_path = preg_replace('/\/+/', '/', $original_path); // Removing leading and trailing slashes. $original_path = preg_replace('/^\/|\/+$/', '', $original_path); list($identifier, $path) = explode("|", $original_path, 2); $identifier = trim($identifier); // Replace identifiers provided by modules implementing hook_cb_identifier_values. $values = NULL; $obj = array('title' => $title, 'path' => $path, 'attributes' => $attributes); foreach (module_implements('cb_identifier_values') as $module) { $values = module_invoke($module, 'cb_identifier_values', $identifier, $obj); if (isset($values)) break; } if (isset($values)) { // Ease return values for callbacks. if (!is_array($values)) { $crumbs[] = array('crumb' => $values, 'title' => $title, 'href' => $path); } elseif (isset($values['crumb']) || isset($values['title']) || isset($values['href'])) { $crumbs[] = $values; } else { $crumbs = $values; } } else { // Use original path if no pipe was given. if (!isset($path)) { $path = $original_path; } $options = parse_url($path); $options = array_merge($options, $attributes); $crumbs[] = array( 'crumb' => l($title, $options['path'], $options), 'title' => $title, 'href' => $options['path'], ); } return $crumbs; } /** * Builds a table of identifiers and their behaviors. * * @ingroup themeable */ function theme_custom_breadcrumbs_help_identifiers() { $identifiers = module_invoke_all('cb_identifier_list'); $headers = array(t('Identifier'), t('Behaviour')); $rows = array(); if (!empty($identifiers)) { foreach ($identifiers as $id => $description) { $rows[] = array(check_plain($id), $description); } } else { $rows[] = array( array( 'data' => t('No special identifiers have been defined. You must enable the custom breadcrumbs identifiers module or another module that implements hook_cb_identifier_list and hook_cb_identifier_values to enable this feature.', array('@link' => url('admin/build/modules'))), 'colspan' => 2, ), ); } return theme('table', $headers, $rows, array('class' => 'description')); } /** * Loads the custom breadcrumb from submodule table. * * @param $module * The name of the custom breadcrumbs submodule managing the requested breadcrumb. * @param $table * The name of the table to limit the search to. This only needs to be provided if * the submodule provides breadcrumbs from more than one table. * @param $param * An array of the form 'field' => $value used in the SQL WHERE clause. * * @return * if $param is empty, all breadcrumbs from the table will be returned as an array * otherwise a single breadcrumb object is be returned. */ function custom_breadcrumbs_load_breadcrumbs($module, $table = NULL, $param = array(), $languages = array()) { $breadcrumbs = array(); $bc_info = module_invoke($module, 'cb_breadcrumb_info'); foreach ($bc_info as $info) { if (!isset($table) || ($info['table'] == $table)) { $args = array(); $cond = array(); $sql = "SELECT * FROM {". $info['table'] ."}"; if ($p = !empty($param)) { $sql .= " WHERE "; foreach ($param as $key => $value) { $cond[] = db_escape_string($key) ." = '%s'"; $args[] = $value; } $cond = implode(' AND ', $cond); if ($cond != '') { $sql .= $cond; } } if (!empty($languages)) { $sql .= ($p) ? " AND " : " WHERE "; $sql .= "language IN (". db_placeholders($languages, 'text') .") ORDER BY language ASC"; $args = array_merge($args, $languages); } $result = db_query($sql, $args); while ($breadcrumb = db_fetch_object($result)) { if (!isset($breadcrumb->name)) { $breadcrumb->name = (isset($info['name_constructor'])) ? $info['name_constructor']($breadcrumb) : $breadcrumb->$info['field']; } $breadcrumb->breadcrumb_type = $info['type']; $breadcrumbs[] = $breadcrumb; } } } return $breadcrumbs; } /** * Determines breadcrumb visibility by evaluating PHP code. * * @param $breadcrumb * The breadcrumb object. * @param $objs * An array of objects (node, taxonomy, or view) that can be used in the php code. * * @return * TRUE if the breadcrumb should be displayed, FALSE otherwise. */ function custom_breadcrumbs_is_visible($breadcrumb, $objs = array()) { $visibility = TRUE; if (isset($breadcrumb)) { // Provide access to objects by standard variable names. foreach ($objs as $key => $obj) { $$key = is_object($obj) ? drupal_clone($obj) : $obj; } // Guard against hidden spaces. if (isset($breadcrumb->visibility_php) && ($trimmed = trim($breadcrumb->visibility_php)) != '' ) { ob_start(); $visibility = eval($trimmed); ob_end_clean(); } } return $visibility; } /** * Loads all breadcrumbs from all submodules. * * Current breadcrumbs are held as static variable. * * @param $refresh * If set to TRUE, reload breadcrumbs from database. * * @return * An array of breadcrumb objects. */ function _custom_breadcrumbs_load_all_breadcrumbs($refresh = FALSE) { static $breadcrumbs; if ($refresh || !isset($breadcrumbs)) { $breadcrumbs = array(); foreach (module_implements('cb_breadcrumb_info') as $module) { $more = custom_breadcrumbs_load_breadcrumbs($module); if (!empty($more)) { $breadcrumbs = array_merge($more, $breadcrumbs); } } } return $breadcrumbs; } /** * Sets the breadcrumb trail to match the menu structure. * * This function uses the same approach as in the menu_breadcrumb module. */ function custom_breadcrumbs_set_menu_breadcrumb() { $menu_item = menu_get_item(); $menu_link = menu_link_load(db_result(db_query("SELECT mlid FROM {menu_links} WHERE link_path = '%s'", $menu_item['href']))); $use_menu = $menu_link['menu_name']; $menus = variable_get('custom_breadcrumbs_menu_list', array() ); if (in_array($use_menu, $menus)) { menu_set_active_menu_name($use_menu); return TRUE; } } /** * Implements hook_theme_registry_alter(). */ function custom_breadcrumbs_theme_registry_alter(&$theme_registry) { if (variable_get('custom_breadcrumbs_force_active_trail', FALSE) && !empty($theme_registry['links'])) { global $theme; // Store the existing theme functions. $themes = variable_get('custom_breadcrumbs_menu_theme', array()); $themes[$theme] = array( 'menu_item' => $theme_registry['menu_item']['function'], 'menu_item_link' => $theme_registry['menu_item_link']['function'], ); variable_set('custom_breadcrumbs_menu_theme', $themes); // Replace these with our own functions. We will call the original functions after call these override functions. $theme_registry['links']['function'] = 'custom_breadcrumbs_override_links'; $theme_registry['menu_item_link']['function'] = 'custom_breadcrumbs_theme_menu_item_link'; $theme_registry['menu_item']['function'] = 'custom_breadcrumbs_theme_menu_item'; } } /** * Determines if a link is in the active trail. * * @param $link * A menu link. * * @return * TRUE if the link is in the active trail, FALSE otherwise. */ function custom_breadcrumbs_in_active_trail($link) { if (!isset($link) || !isset($link['href'])) { return FALSE; } $trail = menu_get_active_trail(); if (!isset($trail)) { return FALSE; } foreach ($trail as $step) { if (isset($step['href']) && ($step['href'] == $link['href'] || $step['href'] == drupal_get_path_alias($link['href']))) { return TRUE; } } return FALSE; } function custom_breadcrumbs_override_links($links, $attributes = array('class' => 'links')) { $output = ''; if (count($links) > 0) { $output = ''; $num_links = count($links); $i = 1; foreach ($links as $key => $link) { $class = $key; // Add first, last and active classes to the list of links to help out themers. if ($i == 1) { $class .= ' first'; } if ($i == $num_links) { $class .= ' last'; } if (isset($link['href']) && ($link['href'] == $_GET['q'] || ($link['href'] == '' && drupal_is_front_page()))) { $class .= ' active'; } if (custom_breadcrumbs_in_active_trail($link) && ($link['href'] != '')) { $class .= ' active-trail'; } $output .= ' $class)) .'>'; if (isset($link['href'])) { // Pass in $link as $options, they share the same keys. $output .= l($link['title'], $link['href'], $link); } elseif (!empty($link['title'])) { // Some links are actually not links, but we wrap these in for adding title and class attributes. if (empty($link['html'])) { $link['title'] = check_plain($link['title']); } $span_attributes = ''; if (isset($link['attributes'])) { $span_attributes = drupal_attributes($link['attributes']); } $output .= ''. $link['title'] .''; } $i++; $output .= "\n"; } $output .= ''; } return $output; } /* code cribbed from dhtml - modified to suit custom breadcrumbs */ /** * Preprocessor for menu_item_link. * * Adds an ID attribute to menu links and helps the module * follow the recursion of menu_tree_output(). * * @param $link * A menu link. */ function custom_breadcrumbs_theme_menu_item_link($link) { // Find out which theme function to dispatch to after preprocessing. global $theme; static $function; if (!isset($function)) { $registry = variable_get('custom_breadcrumbs_menu_theme', array()); $function = isset($registry[$theme]) ? $registry[$theme]['menu_item_link'] : 'theme_menu_item_link'; } if (isset($link['mlid'])) { // Some themes use options, others use localized_options. Populate both. $link['localized_options']['attributes']['id'] = 'custom_breadcrumbs_menu-'. _custom_breadcrumbs_menu_unique_id($link['mlid']); $link['options']['attributes']['id'] = $link['localized_options']['attributes']['id']; // Each link in series is another level of recursion. Add it to the stack. _custom_breadcrumbs_menu_stack($link); if (custom_breadcrumbs_in_active_trail($link)) { $link['localized_options']['attributes']['class'] = 'active'; } } // Pass the altered variables to the normal menu themer. return $function($link); } function custom_breadcrumbs_theme_menu_item($link, $has_children, $menu = '', $in_active_trail = FALSE, $extra_class = NULL) { global $theme; static $function; if (!isset($function)) { $registry = variable_get('custom_breadcrumbs_menu_theme', array()); $function = isset($registry[$theme]) ? $registry[$theme]['menu_item'] : 'theme_menu_item'; } /* When theme('menu_item') is called, the menu tree below it has been * rendered already. Since we are done on this recursion level, * one element must be popped off the stack. */ $item = _custom_breadcrumbs_menu_stack(); // If there are children, but they were not loaded... if ($has_children && !$menu) { // Load the tree below the current position. $tree = _custom_breadcrumbs_menu_subtree($item); $force_active_trail = FALSE; if (!empty($tree)) { foreach ($tree as $sub => $data) { if (custom_breadcrumbs_in_active_trail($data['link'])) { $force_active_trail = TRUE; } else { $belows = (array)$data['below']; foreach ($belows as $id => $below) { // Descend... if (custom_breadcrumbs_in_active_trail($below['link'])) { $force_active_trail = TRUE; } else { unset($tree[$sub]['below'][$id]); } } } } } if ($force_active_trail) { // Render it... $menu = menu_tree_output($tree); $in_active_trail = TRUE; } // Sanitize tree. If we found no children, the item has none. if (!$menu) { $has_children = FALSE; } } // If the current item can expand, and is neither saved as open nor in the active trail, close it. if ($menu && !$in_active_trail) { $extra_class .= ' collapsed start-collapsed '; } // Pass the altered variables to the normal menu themer. return $function($link, $has_children, $menu, $in_active_trail, $extra_class); } /** * Traverses the menu tree and returns the sub-tree of the item indicated by the parameter. * * @param $stack * An array of menu item links that are nested in each other in the tree. * * @return * The items below the lowest item in the stack. */ function _custom_breadcrumbs_menu_subtree($item) { static $index = array(); static $indexed = array(); // This looks expensive, but menu_tree_all_data uses static caching. $tree = menu_tree_all_data($item['menu_name']); if (!isset($indexed[$item['menu_name']])) { $index += _custom_breadcrumbs_menu_index($tree); $indexed[$item['menu_name']] = TRUE; } // Traverse the tree. foreach ($index[$item['mlid']]['parents'] as $mlid) { $key = $index[$mlid]['key']; if (!isset($tree[$key])) { return array(); } $tree = $tree[$key]['below']; } $key = $index[$item['mlid']]['key']; return isset($tree[$key]) ? $tree[$key]['below'] : array(); } /** * Indexes the menu tree by mlid. This is needed to identify the items without relying on titles. * * This function is recursive. * * @param $tree * A tree of menu items such as the return value of menu_tree_all_data(). * * @return * An array associating mlid values with the internal keys of the menu tree. */ function _custom_breadcrumbs_menu_index($tree, $ancestors = array(), $parent = NULL) { $index = array(); if ($parent) $ancestors[] = $parent; foreach ($tree as $key => $item) { $index[$item['link']['mlid']] = array( 'key' => $key, 'parents' => $ancestors, ); if (!empty($item['below'])) { $index += _custom_breadcrumbs_menu_index($item['below'], $ancestors, $item['link']['mlid']); } } return $index; } /** * Tracks the ID attributes and add a suffix to make it unique (when necessary). * * @param $id * The link id. * * @return * The link id, rendered unique by a suffix as needed. */ function _custom_breadcrumbs_menu_unique_id($id) { static $ids = array(); if (!isset($ids[$id])) { $ids[$id] = 1; return $id; } else { return $id .'-'. $ids[$id]++; } } /** * Stores the recursion levels. * * @param $link * If a menu item link is passed, it will be appended to the stack. * If none is given, the stack will be returned and popped by one. * * @return * The stack, if no parameter is given. */ function _custom_breadcrumbs_menu_stack($link = FALSE) { static $stack = array(); if ($link) { array_push($stack, $link); } else { return array_pop($stack); } } /** * Retrieves the module weight. * * @param $name * The module name. * * @return * The module weight. */ function _custom_breadcrumbs_get_module_weight($name) { $weight = (int) db_result(db_query("SELECT weight FROM {system} WHERE name = '%s'", $name)); return $weight; } /** * Determines if a text string is php code and if it is, evaluate it. * * @param $text * A potential code snippet to evaluate. * @param $objs * An optional array of objects to make available to the php code snippet. * * @return * If the text string contains a php code snippet, it will be evaluated, and if * the result is an array, it will be returned. Otherwise nothing is returned. */ function extract_php($text, $objs = array()) { if (drupal_substr(trim($text), 0, 5) == ''), '', $text); foreach ($objs as $key => $obj) { $$key = is_object($obj) ? drupal_clone($obj) : $obj; } ob_start(); $output = eval($text); ob_end_clean(); return (is_array($output)) ? $output : NULL; } } /** * Determines if the current path is in the excluded list. * * @return * TRUE if the current path is on the custom breadcrumbs excluded path list, * FALSE otherwise. */ function custom_breadcrumbs_exclude_path() { static $excluded; if (variable_get('custom_breadcrumbs_use_exclude_list', FALSE)) { if (!isset($excluded)) { $excluded = explode(',', variable_get('custom_breadcrumbs_exclude_list', '')); } if (!empty($excluded)) { module_load_include('inc', 'custom_breadcrumbs', 'custom_breadcrumbs_common'); foreach ($excluded as $path) { if (_custom_breadcrumbs_match_path($_REQUEST['q'], trim($path))) { return TRUE; } } } } return FALSE; } /** * Adds optional html identifiers to breadcrumb links. * * @param $part * A postive integer indicating the breadcrumb segment (home crumb = 0). * * @param $bid * The breadcrumb id. * * @return * An associative array containing the HTML attributes to apply to the anchor tag. */ function _custom_breadcrumbs_identifiers_option($part = 0, $bid = NULL) { $options = array('attributes' => array()); $classes = array(); $base = 'custom-breadcrumbs'; if (variable_get('custom_breadcrumbs_home_id', FALSE) && ($part == 0)) { $options['attributes']['id'] = $base .'-home'; } elseif (variable_get('custom_breadcrumbs_parts_class', FALSE) && ($part > 0)) { $classes[] = $base .'-item-'. $part; } if (variable_get('custom_breadcrumbs_even_odd_class', FALSE)) { $classes[] = ($part % 2 == 0) ? 'even' : 'odd'; } if (!empty($classes)) { $options['attributes']['class'] = implode(' ', $classes); } return $options; }