'Add menu block', 'description' => 'Add a new menu block.', 'access arguments' => array('administer blocks'), 'page callback' => 'drupal_get_form', 'page arguments' => array('menu_block_add_block_form'), 'type' => MENU_LOCAL_TASK, 'file' => 'menu_block.admin.inc', ); $items['admin/build/block/delete-menu-block'] = array( 'title' => 'Delete menu block', 'access arguments' => array('administer blocks'), 'page callback' => 'drupal_get_form', 'page arguments' => array('menu_block_delete'), 'type' => MENU_CALLBACK, 'file' => 'menu_block.admin.inc', ); $items['admin/settings/menu_block'] = array( 'title' => 'Menu block', 'description' => 'Configure menu block.', 'access arguments' => array('administer blocks'), 'page callback' => 'drupal_get_form', 'page arguments' => array('menu_block_admin_settings_form'), 'type' => MENU_NORMAL_ITEM, 'file' => 'menu_block.admin.inc', ); return $items; } /** * Implements hook_help(). */ function menu_block_help($path, $arg) { switch ($path) { case 'admin/build/block/configure': if ($arg[4] != 'menu_block') { break; } case 'admin/help#menu_block': case 'admin/build/block': case 'admin/build/block/add-menu-block': module_load_include('inc', 'menu_block', 'menu_block.pages'); return _menu_block_help($path, $arg); } } /** * Implements hook_theme(). */ function menu_block_theme(&$existing, $type, $theme, $path) { // Add theme hook suggestion patterns for the core theme functions used in // this module. We can't add them during hook_theme_registry_alter() because // we will already have missed the opportunity for the theme engine's // theme_hook() to process the pattern. And we can't run the pattern ourselves // because we aren't given the type, theme and path in that hook. $existing['menu_tree']['pattern'] = 'menu_tree__'; $existing['menu_item']['pattern'] = 'menu_item__'; $existing['menu_item_link']['pattern'] = 'menu_item_link__'; return array( 'menu_block_wrapper' => array( 'template' => 'menu-block-wrapper', 'arguments' => array('content' => NULL, 'settings' => NULL, 'delta' => NULL), 'pattern' => 'menu_block_wrapper__', ), 'menu_block_menu_order' => array( 'arguments' => array('element' => NULL), 'file' => 'menu_block.admin.inc', ), ); } /** * Implements hook_ctools_plugin_directory(). */ function menu_block_ctools_plugin_directory($module, $plugin) { if ($plugin == 'content_types') { return 'plugins/' . $plugin; } } /** * Process variables for menu-block-wrapper.tpl.php. * * @see menu-block-wrapper.tpl.php */ function template_preprocess_menu_block_wrapper(&$variables) { $variables['classes_array'][] = 'menu-block-' . $variables['delta']; $variables['classes_array'][] = 'menu-name-' . $variables['settings']['menu_name']; $variables['classes_array'][] = 'parent-mlid-' . $variables['settings']['parent_mlid']; $variables['classes_array'][] = 'menu-level-' . $variables['settings']['level']; $variables['classes'] = check_plain(implode(' ', $variables['classes_array'])); $variables['template_files'][] = 'menu-block-wrapper-' . $variables['settings']['menu_name']; } /** * Alters the block admin form to add delete links next to menu blocks. */ function menu_block_form_block_admin_display_form_alter(&$form, $form_state) { module_load_include('inc', 'menu_block', 'menu_block.admin'); _menu_block_form_block_admin_display_form_alter($form, $form_state); } /** * Returns a list of menu names implemented by all modules. * * @return * array A list of menu names and titles. */ function menu_block_get_all_menus() { static $all_menus; if (!$all_menus) { // Include book support. if (module_exists('book')) { module_load_include('inc', 'menu_block', 'menu_block.book'); } // We're generalizing menu's menu_get_menus() by making it into a hook. // Retrieve all the menu names provided by hook_get_menus(). $all_menus = module_invoke_all('get_menus'); // Add an option to use the menu for the active menu item. $all_menus[MENU_TREE__CURRENT_PAGE_MENU] = '<' . t('the menu selected by the page') . '>'; asort($all_menus); } return $all_menus; } /** * Implements hook_block(). */ function menu_block_block($op = 'list', $delta = NULL, $edit = NULL) { $function = '_menu_block_block_' . $op; if (function_exists($function)) { return $function($delta, $edit); } else { // "op"s besides "view" are seldom used, so we store them in a separate file. module_load_include('inc', 'menu_block', 'menu_block.admin'); if (function_exists($function)) { return $function($delta, $edit); } } } /** * Returns the configuration for the requested block delta. * * @param $delta * string The delta that uniquely identifies the block in the block system. If * not specified, the default configuration will be returned. * @return * array An associated array of configuration options. */ function menu_block_get_config($delta = NULL) { $config = array( 'delta' => $delta, 'menu_name' => 'primary-links', 'parent_mlid' => 0, 'title_link' => 0, 'admin_title' => '', 'level' => 1, 'follow' => 0, 'depth' => 0, 'expanded' => 0, 'sort' => 0, ); // Get the block configuration options. if ($delta) { $config['title_link'] = variable_get("menu_block_{$delta}_title_link", $config['title_link']); $config['admin_title'] = variable_get("menu_block_{$delta}_admin_title", $config['admin_title']); $config['level'] = variable_get("menu_block_{$delta}_level", $config['level']); $config['follow'] = variable_get("menu_block_{$delta}_follow", $config['follow']); $config['depth'] = variable_get("menu_block_{$delta}_depth", $config['depth']); $config['expanded'] = variable_get("menu_block_{$delta}_expanded", $config['expanded']); $config['sort'] = variable_get("menu_block_{$delta}_sort", $config['sort']); list($config['menu_name'], $config['parent_mlid']) = explode(':', variable_get("menu_block_{$delta}_parent", $config['menu_name'] . ':' . $config['parent_mlid'])); } return $config; } /** * Returns the 'view' $op info for hook_block(). * * @param $delta * string The name of the block to render. */ function _menu_block_block_view($delta) { return menu_tree_build(menu_block_get_config($delta)); } /** * Build a menu tree based on the provided configuration. * * @param $config * array An array of configuration options that specifies how to build the * menu tree and its title. * - delta: (string) The menu_block's block delta. * - menu_name: (string) The machine name of the requested menu. Can also be * set to MENU_TREE__CURRENT_PAGE_MENU to use the menu selected by the page. * - parent_mlid: (int) The mlid of the item that should root the tree. Use 0 * to use the menu's root. * - title_link: (boolean) Specifies if the title should be rendered as a link * or a simple string. * - admin_title: (string) An optional title to uniquely identify the block on * the administer blocks page. * - level: (int) The starting level of the tree. * - follow: (string) Specifies if the starting level should follow the * active menu item. Should be set to 0, 'active' or 'child'. * - depth: (int) The maximum depth the tree should contain, relative to the * starting level. * - expanded: (boolean) Specifies if the entire tree be expanded or not. * - sort: (boolean) Specifies if the tree should be sorted with the active * trail at the top of the tree. * @return * array An array containing the rendered tree in the 'content' key and the * rendered title in the 'subject' key. */ function menu_tree_build($config) { // Retrieve the active menu item from the database. if ($config['menu_name'] == MENU_TREE__CURRENT_PAGE_MENU) { // Retrieve the list of available menus. $menu_order = variable_get('menu_block_menu_order', array('primary-links' => '', 'secondary-links' => '')); // Retrieve all the menus containing a link to the current page. $result = db_query("SELECT menu_name FROM {menu_links} WHERE link_path = '%s'", $_GET['q'] ? $_GET['q'] : ''); while ($item = db_fetch_array($result)) { // Check if the menu is in the list of available menus. if (isset($menu_order[$item['menu_name']])) { // Mark the menu. $menu_order[$item['menu_name']] = MENU_TREE__CURRENT_PAGE_MENU; } } // Find the first marked menu. $config['menu_name'] = array_search(MENU_TREE__CURRENT_PAGE_MENU, $menu_order); $config['parent_mlid'] = 0; // If no menu link was found, don't display the block. if (empty($config['menu_name'])) { return array(); } } // Get the default block name. $menu_names = menu_block_get_all_menus(); menu_block_set_title(t($menu_names[$config['menu_name']])); if ($config['expanded'] || $config['parent_mlid']) { // Get the full, un-pruned tree. $tree = menu_tree_all_data($config['menu_name']); // And add the active trail data back to the full tree. menu_tree_add_active_path($tree); } else { // Get the tree pruned for just the active trail. $tree = menu_tree_page_data($config['menu_name']); } // Allow other modules to alter the tree before we begin operations on it. $alter_data = &$tree; // Also allow modules to alter the config. $alter_data['__drupal_alter_by_ref'] = array(&$config); drupal_alter('menu_block_tree', $alter_data); // Localize the tree. if (module_exists('i18nmenu')) { i18nmenu_localize_tree($tree); } // Prune the tree along the active trail to the specified level. if ($config['level'] > 1 || $config['parent_mlid']) { if ($config['parent_mlid']) { $parent_item = menu_link_load($config['parent_mlid']); menu_tree_prune_tree($tree, $config['level'], $parent_item); } else { menu_tree_prune_tree($tree, $config['level']); } } // Prune the tree to the active menu item. if ($config['follow']) { menu_tree_prune_active_tree($tree, $config['follow']); } // If the menu-item-based tree is not "expanded", trim the tree to the active path. if ($config['parent_mlid'] && !$config['expanded']) { menu_tree_trim_active_path($tree); } // Trim the branches that extend beyond the specified depth. if ($config['depth'] > 0) { menu_tree_depth_trim($tree, $config['depth']); } // Sort the active path to the top of the tree. if ($config['sort']) { menu_tree_sort_active_path($tree); } // Render the tree. $data = array(); $data['subject'] = menu_block_get_title($config['title_link'], $config); $data['content'] = menu_block_tree_output($tree, $config); if ($data['content']) { $hooks = array(); $hooks[] = 'menu_block_wrapper__' . $config['delta']; $hooks[] = 'menu_block_wrapper__' . str_replace('-', '_', $config['menu_name']); $hooks[] = 'menu_block_wrapper'; $data['content'] = theme($hooks, $data['content'], $config, $config['delta']); } return $data; } /** * Retrieves the menu item to use for the tree's title. * * @param $render_title_as_link * boolean A boolean that says whether to render the title as a link or a * simple string. * @return * string The tree's title rendered as a string. */ function menu_block_get_title($render_title_as_link = TRUE, $config = array()) { $menu_item = menu_block_set_title(); // The tree's title is a menu title, a normal string. if (is_string($menu_item)) { $title = check_plain($menu_item); } // The tree's title is a menu item with a link. elseif ($render_title_as_link) { if (!empty($menu_item['in_active_trail'])) { if (!empty($menu_item['localized_options']['attributes']['class'])) { $menu_item['localized_options']['attributes']['class'] .= ' active-trail'; } else { $menu_item['localized_options']['attributes']['class'] = 'active-trail'; } } $hooks = array(); if (!empty($config['delta'])) { $hooks[] = 'menu_item_link__menu_block__' . $config['delta']; } $hooks[] = 'menu_item_link__menu_block__' . str_replace('-', '_', $menu_item['menu_name']); $hooks[] = 'menu_item_link__menu_block'; $hooks[] = 'menu_item_link'; $title = theme($hooks, $menu_item); } // The tree's title is a menu item. else { $title = check_plain($menu_item['title']); } return $title; } /** * Sets the menu item to use for the tree's title. * * @param $item * array The menu item (an array) or the menu item's title as a string. */ function menu_block_set_title($item = NULL) { static $menu_item; // Save the menu item. if (!is_null($item)) { $menu_item = $item; } return $menu_item; } /** * Add the active trail indicators into the tree. * * The data returned by menu_tree_page_data() has link['in_active_trail'] set to * TRUE for each menu item in the active trail. The data returned from * menu_tree_all_data() does not contain the active trail indicators. This is a * helper function that adds it back in. * * @param $tree * array The menu tree. * @return * void */ function menu_tree_add_active_path(&$tree) { // Grab any menu item to find the menu_name for this tree. $menu_item = current($tree); $tree_with_trail = menu_tree_page_data($menu_item['link']['menu_name']); // To traverse the original tree down the active trail, we use a pointer. $subtree_pointer =& $tree; // Find each key in the active trail. while ($tree_with_trail) { foreach (array_keys($tree_with_trail) AS $key) { if ($tree_with_trail[$key]['link']['in_active_trail']) { // Set the active trail info in the original tree. $subtree_pointer[$key]['link']['in_active_trail'] = TRUE; // Continue in the subtree, if it exists. $tree_with_trail =& $tree_with_trail[$key]['below']; $subtree_pointer =& $subtree_pointer[$key]['below']; break; } else { unset($tree_with_trail[$key]); } } } } /** * Trim everything but the active trail in the tree. * * @param $tree * array The menu tree to trim. * @return * void */ function menu_tree_trim_active_path(&$tree) { foreach (array_keys($tree) AS $key) { if (($tree[$key]['link']['in_active_trail'] || $tree[$key]['link']['expanded']) && $tree[$key]['below']) { // Continue in the subtree, if it exists. menu_tree_trim_active_path($tree[$key]['below']); } else { // Trim anything not expanded or along the active trail. $tree[$key]['below'] = FALSE; } } } /** * Sort the active trail to the top of the tree. * * @param $tree * array The menu tree to sort. * @return * void */ function menu_tree_sort_active_path(&$tree) { module_load_include('inc', 'menu_block', 'menu_block.sort'); _menu_tree_sort_active_path($tree); } /** * Prune a tree so that it begins at the specified level. * * This function will follow the active menu trail to the specified level. * * @param $tree * array The menu tree to prune. * @param $level * int The level of the original tree that will start the pruned tree. * @param $parent_item * array The menu item that should be used as the root of the tree. * @return * void */ function menu_tree_prune_tree(&$tree, $level, $parent_item = FALSE) { if (!empty($parent_item)) { // Prune the tree along the path to the menu item. for ($i = 1; $i <= MENU_MAX_DEPTH && $parent_item["p$i"] != '0'; $i++) { $plid = $parent_item["p$i"]; $found_active_trail = FALSE; // Examine each element at this level for the ancestor. foreach (array_keys($tree) AS $key) { if ($tree[$key]['link']['mlid'] == $plid) { menu_block_set_title($tree[$key]['link']); // Prune the tree to the children of this ancestor. $tree = $tree[$key]['below'] ? $tree[$key]['below'] : array(); $found_active_trail = TRUE; break; } } // If we don't find the ancestor, bail out. if (!$found_active_trail) { $tree = array(); break; } } } // Trim the upper levels down to the one desired. for ($i = 1; $i < $level; $i++) { $found_active_trail = FALSE; // Examine each element at this level for the active trail. foreach (array_keys($tree) AS $key) { if ($tree[$key]['link']['in_active_trail']) { // Get the title for the pruned tree. menu_block_set_title($tree[$key]['link']); // Prune the tree to the children of the item in the active trail. $tree = $tree[$key]['below'] ? $tree[$key]['below'] : array(); $found_active_trail = TRUE; break; } } // If we don't find the active trail, the active item isn't in the tree we want. if (!$found_active_trail) { $tree = array(); break; } } } /** * Prune a tree so that it begins at the active menu item. * * @param $tree * array The menu tree to prune. * @param $level * string The level which the tree will be pruned to: 'active' or 'child'. * @return * void */ function menu_tree_prune_active_tree(&$tree, $level) { module_load_include('inc', 'menu_block', 'menu_block.follow'); _menu_tree_prune_active_tree($tree, $level); } /** * Prune a tree so it does not extend beyond the specified depth limit. * * @param $tree * array The menu tree to prune. * @param $depth_limit * int The maximum depth of the returned tree; must be a positive integer. * @return * void */ function menu_tree_depth_trim(&$tree, $depth_limit) { // Prevent invalid input from returning a trimmed tree. if ($depth_limit < 1) { return; } // Examine each element at this level to find any possible children. foreach (array_keys($tree) AS $key) { if ($tree[$key]['below']) { if ($depth_limit > 1) { menu_tree_depth_trim($tree[$key]['below'], $depth_limit-1); } else { // Remove the children items. $tree[$key]['below'] = FALSE; } } if ($depth_limit == 1 && $tree[$key]['link']['has_children']) { // Turn off the menu styling that shows there were children. $tree[$key]['link']['has_children'] = FALSE; $tree[$key]['link']['leaf_has_children'] = TRUE; } } } /** * Returns a rendered menu tree. * * This is an optimized version of menu_tree_output() with additional classes * added to the output. * * @param $tree * array A data structure representing the tree as returned from menu_tree_data. * @return * string The rendered HTML of that data structure. */ function menu_block_tree_output(&$tree, $config = array()) { $output = ''; $items = array(); // Create context if no config was provided. if (empty($config)) { $config['delta'] = 0; // Grab any menu item to find the menu_name for this tree. $menu_item = current($tree); $config['menu_name'] = $menu_item['link']['menu_name']; } $hook_delta = $config['delta']; $hook_menu_name = str_replace('-', '_', $config['menu_name']); // Pull out just the menu items we are going to render so that we // get an accurate count for the first/last classes. foreach (array_keys($tree) as $key) { if (!$tree[$key]['link']['hidden']) { $items[$key] = array( 'link' => $tree[$key]['link'], // To prevent copying the entire child array, we render it first. 'below' => !empty($tree[$key]['below']) ? menu_block_tree_output($tree[$key]['below'], $config) : '', ); } } $num_items = count($items); $i = 1; foreach (array_keys($items) as $key) { // Render the link. $link_class = array(); if (!empty($items[$key]['link']['localized_options']['attributes']['class'])) { $link_class[] = $items[$key]['link']['localized_options']['attributes']['class']; } if ($items[$key]['link']['in_active_trail']) { $link_class[] = 'active-trail'; } if (!empty($link_class)) { $items[$key]['link']['localized_options']['attributes']['class'] = implode(' ', $link_class); } $hooks = array(); $hooks[] = 'menu_item_link__menu_block__' . $hook_delta; $hooks[] = 'menu_item_link__menu_block__' . $hook_menu_name; $hooks[] = 'menu_item_link__menu_block'; $hooks[] = 'menu_item_link'; $link = theme($hooks, $items[$key]['link']); // Render the menu item. $extra_class = array(); if ($i == 1) { $extra_class[] = 'first'; } if ($i == $num_items) { $extra_class[] = 'last'; } $extra_class[] = 'menu-mlid-' . $items[$key]['link']['mlid']; if (!empty($items[$key]['link']['leaf_has_children'])) { $extra_class[] = 'has-children'; } if ($items[$key]['link']['href'] == $_GET['q'] || ($items[$key]['link']['href'] == '' && drupal_is_front_page())) { $extra_class[] = 'active'; } $extra_class = !empty($extra_class) ? implode(' ', $extra_class) : NULL; $hooks = array(); $hooks[] = 'menu_item__menu_block__' . $hook_delta; $hooks[] = 'menu_item__menu_block__' . $hook_menu_name; $hooks[] = 'menu_item__menu_block'; $hooks[] = 'menu_item'; $output .= theme($hooks, $link, $items[$key]['link']['has_children'], $items[$key]['below'], $items[$key]['link']['in_active_trail'], $extra_class); $i++; } $hooks = array(); $hooks[] = 'menu_tree__menu_block__' . $hook_delta; $hooks[] = 'menu_tree__menu_block__' . $hook_menu_name; $hooks[] = 'menu_tree__menu_block'; $hooks[] = 'menu_tree'; return $output ? theme($hooks, $output) : ''; }