导航

解析_theme_build_registry()和_theme_process_registry()

Posted on 2014-05-05 15:30  eastson  阅读(312)  评论(0编辑  收藏  举报

Drupal使用_theme_build_registry()和_theme_process_registry()两个函数构建theme registry。theme registry是theme hook的集合组数。这里以practice模块定义两个theme hook为例,说明一下theme registry的构建过程。

环境:
1. cool_breadcrumbs:定义在practice_theme()中,用function实现。
2. cool_messages:定义在practice_theme()中,用template实现。
3. 使用默认PHPTemplate主题引擎。
4. 使用sub_bartik主题,该主题继承自bartik。

_theme_build_registry()的函数原型:

function _theme_build_registry($theme, $base_theme, $theme_engine) {
  ... ...
}

_theme_build_registry()调用了_theme_process_registry(),而且还调用了很多次:
1. 所有实现了hook_theme()钩子的模块,每个模块都要调用一次:

foreach (module_implements('theme') as $module) {
  // 注意调用时$name=模块名称, $type='module', $theme=模块名称, $path=模块路径
  _theme_process_registry($cache, $module, 'module', $module, drupal_get_path('module', $module));
}

2. 每个base theme都要调用一次。Drupal中的theme是可以继承的,而且可以多级继承。

// Process each base theme.
foreach ($base_theme as $base) {
  // If the base theme uses a theme engine, process its hooks.
  $base_path = dirname($base->filename);
  if ($theme_engine) {
    // 注意调用时$name=theme engine name, $type='base_theme_engine', $theme=base theme name, $path=base theme path
    // 对theme engine调用时,$theme都是对应的theme name, $path都是对应的theme path
    _theme_process_registry($cache, $theme_engine, 'base_theme_engine', $base->name, $base_path);
  }
  // 注意调用时$name=base theme name, $type='base_theme', $theme=base theme name, $path=base theme path
  _theme_process_registry($cache, $base->name, 'base_theme', $base->name, $base_path);
}

3. 对theme engine调用一次。

// And then the same thing, but for the theme.
if ($theme_engine) {
  // 注意调用时$name=theme engine name, $type='theme_engine', $theme=theme name, $path=theme path
  // 对theme engine调用时,$theme都是对应的theme name, $path都是对应的theme path
  _theme_process_registry($cache, $theme_engine, 'theme_engine', $theme->name, dirname($theme->filename));
}

4. 对theme调用一次。

// Finally, hooks provided by the theme itself.
// 注意调用时$name=theme name, $type='theme', $theme=theme name, $path=theme path
_theme_process_registry($cache, $theme->name, 'theme', $theme->name, dirname($theme->filename));


_theme_process_registry()的函数原型:

function _theme_process_registry(&$cache, $name, $type, $theme, $path) {
  ... ... // 注意这里的参数$cache是传址的
}

从上面的_theme_build_registry()知道,_theme_build_registry()会在多种情况下被执行:
1.module
2. base theme engine
3. base theme
4. theme engine
5. theme
每种情况被执行时_theme_build_registry()都会找是否有hook_theme()钩子存在,在hook_theme返回的结果被处理后,合并到$cache,注意$cache参数是传址的。

$variable_process_phases = array(
  'preprocess functions' => 'preprocess',
  'process functions'    => 'process',
);

$hook_defaults = array(
  'variables' => TRUE,
  'render element' => TRUE,
  'pattern' => TRUE,
  'base hook' => TRUE,
);

// Invoke the hook_theme() implementation, process what is returned, and
// merge it into $cache.
$function = $name . '_theme';
if (function_exists($function)) {
  $result = $function($cache, $type, $theme, $path);
  foreach ($result as $hook => $info) {
    // When a theme or engine overrides a module's theme function
    // $result[$hook] will only contain key/value pairs for information being
    // overridden.  Pull the rest of the information from what was defined by
    // an earlier hook.

    // Fill in the type and path of the module, theme, or engine that
    // implements this theme function.
    $result[$hook]['type'] = $type; 
    $result[$hook]['theme path'] = $path;

    // 注意这里的type和theme path,这和传入的参数密切相关
    // 当type=module时,theme path=module path
    // 当type=base theme engine时,因为传入的是base theme path,所以theme path=base theme path
    // 当type=base them时,theme path=base theme path
    // 当type=theme engine时,因为传入的是current theme path,所以theme path=current theme path
    // 当type=theme时,theme path=current theme path

    // If function and file are omitted, default to standard naming
    // conventions.
    if (!isset($info['template']) && !isset($info['function'])) {
      $result[$hook]['function'] = ($type == 'module' ? 'theme_' : $name . '_') . $hook;
    }

    // 在hook_theme()定义theme hook时,一般会设置template或者function
    // 如果两者都没有设置,则默认地type=module时为theme_$hook(),或者$name_$hook()
    // 例如type=theme,则可能是bartik_cool_messages()
    // 或者type=theme engine,则可能是phptemplate_cool_engine()

    if (isset($cache[$hook]['includes'])) {
      $result[$hook]['includes'] = $cache[$hook]['includes'];
    }

    // $cache[$hook]是什么意思?
    // 在后面会看到这样一句代码:$cache = $result + $cache;
    // $cache合并了当前的hook_theme()结果$result,并确保$result优先$cache
    // 举例来说,可能在practice模块中定义了cool_messsages,
    // 当_theme_process_registry()在type=module被执行完成后,$cache就应该有了cool_messages。
    // 后面的bartik又修改了cool_messages的定义,当type=theme执行完成后,
    // $cache的就是两者合并后的结果,并且后面定义的优先于前面定义的。

    // If the theme implementation defines a file, then also use the path
    // that it defined. Otherwise use the default path. This allows
    // system.module to declare theme functions on behalf of core .include
    // files.
    if (isset($info['file'])) {
      $include_file = isset($info['path']) ? $info['path'] : $path;
      $include_file .= '/' . $info['file'];
      include_once DRUPAL_ROOT . '/' . $include_file;
      $result[$hook]['includes'][] = $include_file;
    }

    // 可以定义一个专门的theme function file,用$info['file']说明
    // 这个theme function file可能不属于当前模块,例如可能是system模块用到core模块里面的文件,
    // 这时候需要用$info['path']说明这个theme function file所在的目录

    // If the default keys are not set, use the default values registered
    // by the module.
    if (isset($cache[$hook])) {
      $result[$hook] += array_intersect_key($cache[$hook], $hook_defaults);
    }

    // The following apply only to theming hooks implemented as templates.
    if (isset($info['template'])) {
      // Prepend the current theming path when none is set.
      if (!isset($info['path'])) {
        // 将template转成完整路径
        $result[$hook]['template'] = $path . '/' . $info['template'];
      }
    }

    // Allow variable processors for all theming hooks, whether the hook is
    // implemented as a template or as a function.
    foreach ($variable_process_phases as $phase_key => $phase) {
      // Check for existing variable processors. Ensure arrayness.
      if (!isset($info[$phase_key]) || !is_array($info[$phase_key])) {
        $info[$phase_key] = array();
        $prefixes = array();
        if ($type == 'module') {
          // Default variable processor prefix.
          $prefixes[] = 'template';
          // Add all modules so they can intervene with their own variable
          // processors. This allows them to provide variable processors even
          // if they are not the owner of the current hook.
          $prefixes += module_list();
        }
        elseif ($type == 'theme_engine' || $type == 'base_theme_engine') {
          // Theme engines get an extra set that come before the normally
          // named variable processors.
          $prefixes[] = $name . '_engine';
          // The theme engine registers on behalf of the theme using the
          // theme's name.
          $prefixes[] = $theme;
        }
        else {
          // This applies when the theme manually registers their own variable
          // processors.
          $prefixes[] = $name;
        }
        foreach ($prefixes as $prefix) {
          // Only use non-hook-specific variable processors for theming hooks
          // implemented as templates. See theme().
          if (isset($info['template']) && function_exists($prefix . '_' . $phase)) {
            $info[$phase_key][] = $prefix . '_' . $phase;
          }
          if (function_exists($prefix . '_' . $phase . '_' . $hook)) {
            $info[$phase_key][] = $prefix . '_' . $phase . '_' . $hook;
          }
        }

        // variable processor,变量处理器,这是一个数组
        // 变量处理器分两种:preprocess function和皮肉process function
        // 当type=module时,按照下列顺序查找(以cool_messages为例):
        //       template_preprocess()
        //       template_preprocess_cool_messages()
        //       moduleA_preprocess()
        //       moduleA_preprocess_cool_messages()
        //       ... ...
        //       moduleZ_preprocess()
        //       moduleZ_preprocess_cool_messages()
        //       ---
        //       template_process()
        //       template_process_cool_messages()
        //       moduleA_process()
        //       moduleA_process_cool_messages()
        //       ... ...
        //       moduleZ_process()
        //       moduleZ_process_cool_messages()
        //
        // 当type=theme_engine/base_theme_engine时,按照下列顺序查找:
        //       phptemplate_engine_preprocess()
        //       phptemplate_engine_preprocess_cool_messages()
        //       bartik_preprocess()
        //       bartik_preprocess_cool_messages()
        //       ---
        //       phptemplate_engine_process()
        //       phptemplate_engine_process_cool_messages()
        //       bartik_process()
        //       bartik_process_cool_messages()
        //
        // 当type=theme/base_theme时,按照下列顺序查找:
        //       bartik_preprocess()
        //       bartik_preprocess_cool_messages()
        //       ---
        //       bartik_process()
        //       bartik_process_cool_messages()

      }
      // Check for the override flag and prevent the cached variable
      // processors from being used. This allows themes or theme engines to
      // remove variable processors set earlier in the registry build.
      if (!empty($info['override ' . $phase_key])) {
        // Flag not needed inside the registry.
        unset($result[$hook]['override ' . $phase_key]);
        // 色泽override preprocess functions这样的参数,
        // 可以忽略其它情况下的变量处理器
      }
      elseif (isset($cache[$hook][$phase_key]) && is_array($cache[$hook][$phase_key])) {
        $info[$phase_key] = array_merge($cache[$hook][$phase_key], $info[$phase_key]);
        // 这里将$cache[$hook][$phase_key]和$info[$phase_key]做了合并,
        // 也就是说可能出现这样的情况:
        //       practice_preprocess()/practice_preproces_cool_messages()    模块practice的缺省处理器
        //       other_preproces_cool_messages()                             模块other定义的cool_messages处理器
        //       phptemplate_engine_preprocess_cool_messages()               主题引擎定义的cool_messages处理器
        //       bartik_preprocess_cool_message()                            主题bartik定义的cool_messages处理器
        //       sub_bartik_preprocess_cool_message()                        主题sub_bartik定义的cool_messages处理器
        // 这些变量处理器是没有顺序的,这一点要特别注意
      }
      $result[$hook][$phase_key] = $info[$phase_key];
    }
  }

  // Merge the newly created theme hooks into the existing cache.
  // 合并$result到$cache,这一句代码很重要
  // _theme_build_registry()执行完成后,$cache就是最后的theme registry
  $cache = $result + $cache;
}

当theme/base theme被执行时,_theme_process_registry()会自动关联在theme里面定义的变量处理器:

// $result是hook_theme()返回的结果
// phptemplate_theme()做了两件事情:
// 一是搜索get_defined_functions()中所有的theme function,
// 二是搜索主题目录下所有以.tpl.php结尾的theme template,
// 然后返回到$result,再保存到$cache.
// 例如,theme目录中可以存在一个cool_sideleft.tpl.php这样一个theme template,
// 不需要用hook_theme()定义这个theme hook,phptemplate会自动搜索这个theme template注册到$cache。
// 
// phptemplate执行在前,theme执行在后,
// 当theme执行时,$cache已经包含了them中定义所有theme function和theme template。
// 这时候,可以在theme定义变量处理器,例如bartik_preprocess_cool_sideleft()。
// 执行下面的代码可以将bartik_preprocess_cool_sideleft()关联到cool_sideleft。

// Let themes have variable processors even if they didn't register a
// template.
if ($type == 'theme' || $type == 'base_theme') {
  foreach ($cache as $hook => $info) {
    // Check only if not registered by the theme or engine.
    if (empty($result[$hook])) {
      foreach ($variable_process_phases as $phase_key => $phase) {
        if (!isset($info[$phase_key])) {
          $cache[$hook][$phase_key] = array();
        }
        // Only use non-hook-specific variable processors for theming hooks
        // implemented as templates. See theme().
        if (isset($info['template']) && function_exists($name . '_' . $phase)) {
          $cache[$hook][$phase_key][] = $name . '_' . $phase;
        }
        if (function_exists($name . '_' . $phase . '_' . $hook)) {
          $cache[$hook][$phase_key][] = $name . '_' . $phase . '_' . $hook;
          $cache[$hook]['theme path'] = $path;
        }
        // Ensure uniqueness.
        $cache[$hook][$phase_key] = array_unique($cache[$hook][$phase_key]);
      }
    }
  }
}


情境A:接文章开头假设的例子,我们用practice_theme()定义两个theme hook:

function practice_theme() {
    return array(
        'cool_breadcrumbs' => array(
            'render element' => 'breadcrumbs',
            'function' => 'practice_cool_breadcrumbs',
        ),
        'cool_messages' => array(
            'render element' => 'messages',
            'template' => 'cool_messages',
        ),
    );
}

要完成上面的定义,需要建立一个名为practice_cool_breadcrumbs的theme function:

function practice_cool_breadcrumbs($variables) {
  return '<div>Demo Breadcrumbs</div>';
}

还需要建立一个名为cool_messages.tpl.php的theme template:

<?php
// sites/all/modules/practice/cool_messages.tpl.php
print '<div>Demo Messages</div>';

到这里,practice模块新增的两个theme hook就完成了。当_theme_process_registry($type='module', $name='practice')执行完成后,对应的$result结果如下:

$result = array(
  'cool_breadcrumbs' => array(
    'render element' => 'breadcrumbs',
    'function' => 'practice_cool_breadcrumbs',
    'type' => 'module',
    'theme path' => 'sites/all/modules/practice',
    'preprocess functions' => array(),
    'process functions' => array(),
  ),
  'cool_messages' => array(
    'render element' => 'messages',
    'template' => 'sites/all/modules/practice/cool_messages', // template转成了完整路径
    'type' => 'module',
    'theme path' => 'sites/all/modules/practice',
    'preprocess functions' => array( // 用template实现的theme hook,默认的会加上template_preprocess/process()
      0 => 'template_preprocess', 
    ),
    'process functions' => array(
      0 => 'template_process',
    ),
  ),
);

因为没有在其它地方修改这两个theme hook,所有最后的$cache和$result是一样的。

情境B:续情境A。theme hook可以定义变量处理器preprocess function和process function。实质上,情境A中的template_preprocess()和template_process()也是变量处理器。

function practice_preprocess(&$variables, $hook) { ... }
function practice_process(&$variables, $hook) { ... }

_theme_process_registry($type='module', $name='practice')执行完成后,$result结果如下:

$result = array(
  'cool_breadcrumbs' => array(
    ... ...
    'preprocess functions' => array(), // 不带theme hook名称的变量处理器不适用于function
    'process functions' => array(),
  ),
  'cool_messages' => array(
    ... ...
    'preprocess functions' => array(
      0 => 'template_preprocess', 
      1 => 'practice_preprocess', // 不带theme hook名称的变量处理器只针对template
    ),
    'process functions' => array(
      0 => 'template_process', 
      1 => 'practice_preprocess',
    ),
  ),
);

上面的变量处理器不带theme hook名称,这样的处理器适用与所有用template实现的theme hook。带名称的处理器只适用于指定名称的theme hook:

function practice_preprocess_cool_breadcrumbs(&$variables, $hook) { ... }

$result = array(
  'cool_breadcrumbs' => array(
    'preprocess functions' => array(
      0 => 'practice_preprocess_cool_breadcrumbs', // 带名称的处理器只适用于指定名称的theme hook
    ),
    'process functions' => array(),
  ),
  ... ...
);

 

情境C:续情境A。情境B是在theme hook定义所在的模块practice建立处理器,在其它模块也是可以的。其它模块定义处理器也可以分通用(不带theme hook名称,适用与所有的template)和专用(带theme hook名称,只适用于指定名称)两种。

function other_preprocess(&$variables, $hook) { ... }
function other_preprocess_cool_messages(&$variables, $hook) { ... }

$result = array(
  'cool_messages' => array(
    'preprocess functions' => array(
      0 => 'template_preprocess',
      1 => 'other_preprocess',
      2 => 'other_preprocess_cool_messages',
    ),
    'process functions' => array(
      0 => 'template_process', 
    ),
  ),
);

 

情境D:续情境A。在theme中也可以为theme hook定义处理器。这部分的处理是在_theme_process_registry($type='theme/base_theme')的最后实现的:

if ($type == 'theme' || $type == 'base_theme') {
  foreach ($cache as $hook => $info) { // 只针对在$cache中已经注册过的theme hook
    // Check only if not registered by the theme or engine.
    if (empty($result[$hook])) { // empty($result[$hook])表明是没有在theme的hook_theme()钩子中定义
      foreach ($variable_process_phases as $phase_key => $phase) {
        if (!isset($info[$phase_key])) {
          $cache[$hook][$phase_key] = array();
        }
        // Only use non-hook-specific variable processors for theming hooks
        // implemented as templates. See theme().
        if (isset($info['template']) && function_exists($name . '_' . $phase)) {
          $cache[$hook][$phase_key][] = $name . '_' . $phase;
        }
        if (function_exists($name . '_' . $phase . '_' . $hook)) {
          $cache[$hook][$phase_key][] = $name . '_' . $phase . '_' . $hook;
          $cache[$hook]['theme path'] = $path;
        }
        // Ensure uniqueness.
        $cache[$hook][$phase_key] = array_unique($cache[$hook][$phase_key]);
      }
    }
  }
}

例如,情境A中的cool_messages,可以在theme中再定义一个处理器:

function bartik_preprocess_cool_messages(&$variables) { ... }

// _theme_process_registry($type='module', $name='practice')执行完毕后的$cache
$cache = array(
  'cool_messages' => array(
    'preprocess functions' => array(
      0 => 'template_preprocess', 
    ),
    'process functions' => array(
      0 => 'template_process',
    ),
  ),
);

// _theme_process_registry($type='theme', $name='bartik')执行完毕后的$cache
$cache = array(
  'cool_messages' => array(
    'preprocess functions' => array(
      0 => 'template_preprocess',
      1 => 'bartik_preprocess_cool_messages', // bartik中定义的处理器
    ),
    'process functions' => array(
      0 => 'template_process',
    ),
  ),
);

 

情境E:续情境A。hook_theme()定义的theme hook可以在主题theme中被重载。这里的重载有几个意思:
1. theme function重载。例如,cool_breadcrumbs默认由practice_cool_breadcrumbs()输出,可以在主题中定义bartik_cool_breadcrumbs()实现重载。
2. theme template重载。例如,cool_messages默认由practice/cool_messages.tpl.php输出,可以在主题中定义cool_messages.tpl.php实现重载。
3. theme suggestion重载。例如,名为page的theme hook,默认可能是在某个模块中顶一顶,可以在主题中定义bartik_page()实现函数重载,定义bartik_page__node()/bartik_page__node__1()这样的函数实现theme suggestion函数重载;也可以在主题中定义page.tpl.php实现模板重载,定义page__node.tpl.php/page__node__1.tpl.php这样的模板实现theme suggestion模板重载。

function phptemplate_theme($existing, $type, $theme, $path) {
  $templates = drupal_find_theme_functions($existing, array($theme));
  $templates += drupal_find_theme_templates($existing, '.tpl.php', $path);
  return $templates;
}

function drupal_find_theme_functions($cache, $prefixes) {
  $implementations = array();
  $functions = get_defined_functions();

  // 查找以theme名称开头的theme functions
  // 注意这里只处理$cache中已注册的theme hook
  // $prefix指的是theme名称, 例如bartik

  foreach ($cache as $hook => $info) { 
    foreach ($prefixes as $prefix) { 

      // Find theme functions that implement possible "suggestion" variants of
      // registered theme hooks and add those as new registered theme hooks.
      // The 'pattern' key defines a common prefix that all suggestions must
      // start with. The default is the name of the hook followed by '__'. An
      // 'base hook' key is added to each entry made for a found suggestion,
      // so that common functionality can be implemented for all suggestions of
      // the same base hook. To keep things simple, deep hierarchy of
      // suggestions is not supported: each suggestion's 'base hook' key
      // refers to a base hook, not to another suggestion, and all suggestions
      // are found using the base hook's pattern, not a pattern from an
      // intermediary suggestion.

      // 这里查找的是theme suggestion functions
      // 例如,在$cahce中已存在名为page的theme hook
      // 可以在theme中定义一系列的theme suggestion functions:
      //     bartik_page__node()
      //     bartik_page__node__1()
      //
      // theme suggestion会做为一个新的theme hook被注册,例如
      //     $cache['page__node__1'] = ...
      //
      // theme suggesion hook的base hook被标记为$hook

      $pattern = isset($info['pattern']) ? $info['pattern'] : ($hook . '__');
      if (!isset($info['base hook']) && !empty($pattern)) {
        $matches = preg_grep('/^' . $prefix . '_' . $pattern . '/', $functions['user']);
        if ($matches) {
          foreach ($matches as $match) {
            $new_hook = substr($match, strlen($prefix) + 1);
            $arg_name = isset($info['variables']) ? 'variables' : 'render element';
            $implementations[$new_hook] = array(
              'function' => $match,
              $arg_name => $info[$arg_name],
              'base hook' => $hook,
            );
          }
        }
      }

      // Find theme functions that implement registered theme hooks and include
      // that in what is returned so that the registry knows that the theme has
      // this implementation.

      // 如果存在bartik_page()函数,则覆盖hook_theme()定义的page实现
      //
      // 覆盖是在_theme_process_registry()处理的, $result是phptemplate_theme()返回的结果:
      // if (isset($cache[$hook])) {
      //  $result[$hook] += array_intersect_key($cache[$hook], $hook_defaults);
      // }

      if (function_exists($prefix . '_' . $hook)) {
        $implementations[$hook] = array(
          'function' => $prefix . '_' . $hook,
        );
      }
    }
  }

  return $implementations;
}

function drupal_find_theme_templates($cache, $extension, $path) {
  $implementations = array();

  // Collect paths to all sub-themes grouped by base themes. These will be
  // used for filtering. This allows base themes to have sub-themes in its
  // folder hierarchy without affecting the base themes template discovery.
  $theme_paths = array();
  foreach (list_themes() as $theme_info) {
    if (!empty($theme_info->base_theme)) {
      $theme_paths[$theme_info->base_theme][$theme_info->name] = dirname($theme_info->filename);
    }
  }
  foreach ($theme_paths as $basetheme => $subthemes) {
    foreach ($subthemes as $subtheme => $subtheme_path) {
      if (isset($theme_paths[$subtheme])) {
        $theme_paths[$basetheme] = array_merge($theme_paths[$basetheme], $theme_paths[$subtheme]);
      }
    }
  }
  global $theme;
  $subtheme_paths = isset($theme_paths[$theme]) ? $theme_paths[$theme] : array();

  // Escape the periods in the extension.
  $regex = '/' . str_replace('.', '\.', $extension) . '$/';
  // Get a listing of all template files in the path to search.
  $files = drupal_system_listing($regex, $path, 'name', 0);

  // 在theme目录下查找所有以.tpl.php结尾的模板文件:
  //    page.tpl.php
  //    page__node.tpl.php
  //    page__node__1.tpl.php

  // Find templates that implement registered theme hooks and include that in
  // what is returned so that the registry knows that the theme has this
  // implementation.
  foreach ($files as $template => $file) {
    // Ignore sub-theme templates for the current theme.
    if (strpos($file->uri, str_replace($subtheme_paths, '', $file->uri)) !== 0) {
      continue;
    }
    // Chop off the remaining extensions if there are any. $template already
    // has the rightmost extension removed, but there might still be more,
    // such as with .tpl.php, which still has .tpl in $template at this point.
    if (($pos = strpos($template, '.')) !== FALSE) {
      $template = substr($template, 0, $pos);
    }
    // Transform - in filenames to _ to match function naming scheme
    // for the purposes of searching.
    $hook = strtr($template, '-', '_');
    if (isset($cache[$hook])) { 
      // 只处理在$cache中注册的theme hook
      // 如果存page.tpl.php模板, 则覆盖hook_theme()定义的page实现
      $implementations[$hook] = array(
        'template' => $template,
        'path' => dirname($file->uri),
      );
    }
  }

  // Find templates that implement possible "suggestion" variants of registered
  // theme hooks and add those as new registered theme hooks. See
  // drupal_find_theme_functions() for more information about suggestions and
  // the use of 'pattern' and 'base hook'.
  $patterns = array_keys($files);
  foreach ($cache as $hook => $info) { // 循环处理在$cache中注册的theme hook
    $pattern = isset($info['pattern']) ? $info['pattern'] : ($hook . '__');
    if (!isset($info['base hook']) && !empty($pattern)) {
      // Transform _ in pattern to - to match file naming scheme
      // for the purposes of searching.
      $pattern = strtr($pattern, '_', '-');

      // 这里查找的是theme suggestion templates
      // theme suggestion会做为一个新的theme hook被注册,例如
      //     $cache['page__node__1'] = ...
      //
      // theme suggesion hook的base hook被标记为$hook

      $matches = preg_grep('/^' . $pattern . '/', $patterns);
      if ($matches) {
        foreach ($matches as $match) {
          $file = substr($match, 0, strpos($match, '.'));
          // Put the underscores back in for the hook name and register this
          // pattern.
          $arg_name = isset($info['variables']) ? 'variables' : 'render element';
          $implementations[strtr($file, '-', '_')] = array(
            'template' => $file,
            'path' => dirname($files[$match]->uri),
            $arg_name => $info[$arg_name],
            'base hook' => $hook,
          );
        }
      }
    }
  }
  return $implementations;
}

 

情境F:独立情境。除了在module中用hook_theme()钩子定义theme hook外,也可以在theme和theme engine用hook_theme()钩子theme hook。

function bartik_theme() {
  return array(
    'cool_sideleft' => array(
      'render element' => 'sideleft',
      'template' => 'cool_sideleft',
    ),
  );
}

function bartik_preprocess_cool_sideleft(&$variables) { ... }

$result = array(
  'cool_sideleft' => array(
    'render element' => 'sideleft',
    'template' => 'themes/bartik/cool_sideleft',
    'type' => 'theme',
    'theme path' => 'themes/bartik',
    'preprocess functions' => array(
      0 => 'bartik_preprocess_cool_sideleft', // 没有template_preprocess(). 处理器要以bartik_开头。
    ),
    'process functions' => array(),
  ),
);