Skip to content
menu.inc 6.04 KiB
Newer Older
Dries Buytaert's avatar
Dries Buytaert committed
<?php
Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * @defgroup menu Menu system
 * @{
 */

define('MENU_SHOW', 0);
define('MENU_HIDE', 1);
define('MENU_HIDE_NOCHILD', 2);

/** @} */
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
 * Register a menu item with the menu system.
Dries Buytaert's avatar
 
Dries Buytaert committed
 *
 * @ingroup menu
 * @param $path Location then menu item refers to.
 * @param $title The title of the menu item to show in the rendered menu.
 * @param $callback The function to call when this is the active menu item.
 * @param $weight Heavier menu items sink down the menu.
 * @param $hidden
 * - MENU_SHOW show the menu (default).
 * - MENU_HIDE hide the menu item, but register a callback.
 * - MENU_HIDE_NOCHILD hide the menu item when it has no children.
Dries Buytaert's avatar
 
Dries Buytaert committed
function menu($path, $title, $callback = NULL, $weight = 0, $hidden = MENU_SHOW) {
Dries Buytaert's avatar
 
Dries Buytaert committed

  // add the menu to the flat list of menu items:
Dries Buytaert's avatar
 
Dries Buytaert committed
  $_list[$path] = array("title" => $title, "callback" => $callback, "weight" => $weight, "hidden" => $hidden);
Dries Buytaert's avatar
 
Dries Buytaert committed
}
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
 * Returns an array with the menu items that lead to the specified path.
Dries Buytaert's avatar
 
Dries Buytaert committed
function menu_get_trail($path) {
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  $trail = array();
Dries Buytaert's avatar
 
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  while ($path) {
    if ($_list[$path]) {
      array_unshift($trail, $path);
Dries Buytaert's avatar
 
Dries Buytaert committed

    $path = substr($path, 0, strrpos($path, "/"));
Dries Buytaert's avatar
 
Dries Buytaert committed
  }

Dries Buytaert's avatar
 
Dries Buytaert committed
  return $trail;
Dries Buytaert's avatar
 
Dries Buytaert committed
}

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Returns the path of the active menu item.
Dries Buytaert's avatar
 
Dries Buytaert committed
 * @ingroup menu
Dries Buytaert's avatar
 
Dries Buytaert committed
 */
function menu_get_active_item() {
Dries Buytaert's avatar
 
Dries Buytaert committed
  return menu_set_active_item();
}

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Sets the path of the active menu item.
 * @ingroup menu
 */
Dries Buytaert's avatar
 
Dries Buytaert committed
function menu_set_active_item($path = NULL) {
Dries Buytaert's avatar
 
Dries Buytaert committed
  static $stored_path;
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  if (is_null($stored_path) || !empty($path)) {
    if (empty($path)) {
      $path = $_GET["q"];
    }
    else {
      $_GET['q'] = $path;
    }
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
    while ($path && !$_list[$path]) {
      $path = substr($path, 0, strrpos($path, "/"));
Dries Buytaert's avatar
 
Dries Buytaert committed
    }
Dries Buytaert's avatar
 
Dries Buytaert committed
    $stored_path = $path;
Dries Buytaert's avatar
Dries Buytaert committed
  }

Dries Buytaert's avatar
 
Dries Buytaert committed
  return $stored_path;
Dries Buytaert's avatar
Dries Buytaert committed
}

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
Dries Buytaert's avatar
 
Dries Buytaert committed
 * Returns the title of the active menu item.
 */
function menu_get_active_title() {
  global $_list;
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  if ($path = menu_get_active_item()) {
    return ucfirst($_list[$path]["title"]);
  }
}
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
Dries Buytaert's avatar
 
Dries Buytaert committed
 * Returns the help associated with the active menu item.
 */
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  if (menu_active_handler_exists()) {
    $path = $_GET["q"];
    $output = "";

    $return = module_invoke_all("help", $path);
    foreach ($return as $item) {
      if (!empty($item)) {
        $output .= $item ."\n";
      }
Dries Buytaert's avatar
 
Dries Buytaert committed
    }
Dries Buytaert's avatar
 
Dries Buytaert committed
    return $output;
Dries Buytaert's avatar
Dries Buytaert committed
  }
Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Returns an array of rendered menu items in the active breadcrumb trail.
 */
function menu_get_active_breadcrumb() {
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  $links[] = l(t("Home"), "");
Dries Buytaert's avatar
 
Dries Buytaert committed
  $trail = menu_get_trail($_GET["q"]);
  foreach ($trail as $item) {
    $links[] = _render_item($item);
Dries Buytaert's avatar
Dries Buytaert committed
  }

Dries Buytaert's avatar
 
Dries Buytaert committed
  return $links;
Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Execute the handler associated with the active menu item.
 */
function menu_execute_active_handler() {
  global $_list;
Dries Buytaert's avatar
 
Dries Buytaert committed
  $path = menu_get_active_item();
Dries Buytaert's avatar
 
Dries Buytaert committed
  if ($_list[$path]["callback"]) {
    $arg = substr($_GET["q"], strlen($path) + 1);
Dries Buytaert's avatar
 
Dries Buytaert committed
    if (isset($arg)) {
      return call_user_func_array($_list[$path]["callback"], explode("/", $arg));
Dries Buytaert's avatar
 
Dries Buytaert committed
    }
    else {
Dries Buytaert's avatar
 
Dries Buytaert committed
      return call_user_func($_list[$path]["callback"]);
Dries Buytaert's avatar
 
Dries Buytaert committed
    }
  }
Dries Buytaert's avatar
Dries Buytaert committed
}

Dries Buytaert's avatar
 
Dries Buytaert committed
function menu_active_handler_exists() {
  global $_list;

  $path = menu_get_active_item();

  return function_exists($_list[$path]["callback"]);
}

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Returns true when the path is in the active trail.
 */
function menu_in_active_trail($path) {
  static $trail;
Dries Buytaert's avatar
 
Dries Buytaert committed
  if (empty($trail)) {
    $trail = menu_get_trail($_GET["q"]);
  }
Dries Buytaert's avatar
 
Dries Buytaert committed
  return in_array($path, $trail);
Dries Buytaert's avatar
 
Dries Buytaert committed
}
Dries Buytaert's avatar
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Returns true when the menu has visisble children.
 */
function menu_has_visible_children($item) {
  global $_list;

  if ($_list[$item]['children']) {
    foreach ($_list[$item]['children'] as $child) {
      if ($_list[$child]['hidden'] == MENU_SHOW) {
        return true;
      }
    }
  }

  return false;
}

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Returns a rendered menu tree.
 */
Dries Buytaert's avatar
 
Dries Buytaert committed
function menu_tree($parent = "", $hidden = 0) {
Dries Buytaert's avatar
 
Dries Buytaert committed
  $output = "";
Dries Buytaert's avatar
 
Dries Buytaert committed

Dries Buytaert's avatar
 
Dries Buytaert committed
  if (empty($trail)) {
Dries Buytaert's avatar
 
Dries Buytaert committed
    $trail = menu_get_trail($_GET["q"]);
Dries Buytaert's avatar
 
Dries Buytaert committed
  if (isset($_list[$parent]) && $_list[$parent]["children"]) {
    usort($_list[$parent]["children"], "_menu_sort");
    foreach ($_list[$parent]["children"] as $item) {
Dries Buytaert's avatar
 
Dries Buytaert committed
      /*
      ** Don't render the menu when it is hidden, or when it has no call-back
      ** nor children.  The latter check avoids that useless links are being
      ** rendered.
      */
Dries Buytaert's avatar
 
Dries Buytaert committed
      $visible = menu_has_visible_children($item);
      if (($_list[$item]["hidden"] == MENU_SHOW && $_list[$item]["callback"]) ||
         ($_list[$item]["hidden"] == MENU_SHOW && $visible) ||
         ($_list[$item]["hidden"] == MENU_HIDE_NOCHILD && $visible)) {
        $style = ($visible ? (menu_in_active_trail($item)  ? "expanded" : "collapsed") : "leaf");
Dries Buytaert's avatar
 
Dries Buytaert committed
        $output .= "<li class=\"$style\">";
Dries Buytaert's avatar
 
Dries Buytaert committed
        $output .= _render_item($item);
Dries Buytaert's avatar
 
Dries Buytaert committed
        if (menu_in_active_trail($item)) {
Dries Buytaert's avatar
 
Dries Buytaert committed
          $output .= menu_tree($item);
        }
        $output .= "</li>\n";
      }
Dries Buytaert's avatar
 
Dries Buytaert committed
      else if ($_list[$item]["hidden"] == MENU_HIDE && $_list[$item]["children"]) {
Dries Buytaert's avatar
 
Dries Buytaert committed
        $output .= menu_tree($item, 1);
      }
Dries Buytaert's avatar
 
Dries Buytaert committed
    }
Dries Buytaert's avatar
 
Dries Buytaert committed
    if ($output != '' && $hidden != MENU_HIDE) {
Dries Buytaert's avatar
Dries Buytaert committed
  }

Dries Buytaert's avatar
 
Dries Buytaert committed
  return $output;
Dries Buytaert's avatar
Dries Buytaert committed
}

Dries Buytaert's avatar
 
Dries Buytaert committed
/**
 * Query to module to build the menu.
 */
Dries Buytaert's avatar
 
Dries Buytaert committed
function menu_build($type) {
  /*
  ** Build a sequential list of all menus items.
  */
Dries Buytaert's avatar
 
Dries Buytaert committed

  module_invoke_all("link", $type);

  /*
  ** Tree-ify the sequential list of menu items by adding each
  ** menu item to the 'children' array of their direct parent.
  */

  global $_list;

  foreach ($_list as $path => $data) {

    /*
    ** Find $path's direct parent:
    */
    $parent = $path;
    do {
      $parent = substr($parent, 0, strrpos($parent, "/"));
    }
    while ($parent && !$_list[$parent]);

    if ($path) {
      $_list[$parent]["children"][] = $path;
    }
  }
Dries Buytaert's avatar
 
Dries Buytaert committed
}

Dries Buytaert's avatar
 
Dries Buytaert committed
function _menu_sort($a, $b) {
  global $_list;

  $a = &$_list[$a];
  $b = &$_list[$b];

  return $a["weight"] < $b["weight"] ? -1 : ($a["weight"] > $b["weight"] ? 1 : ($a["title"] < $b["title"] ? -1 : 1));
}

function _render_item($path) {
  global $_list;

Dries Buytaert's avatar
 
Dries Buytaert committed
  return l($_list[$path]["title"], $path);
Dries Buytaert's avatar
 
Dries Buytaert committed
}


Dries Buytaert's avatar
 
Dries Buytaert committed
?>