Drupal 模块开发基本教程(二)
2012-11-29 14:19
232 查看
第二部分:模块的自定义页面显示方法
许多时候我们需要为一些数据显示一个自定义格式的页面。熟悉模板的同志们可能曾经失望的发现,模板只能控制除$content之外的那部分页面。在模板里,内容区之外的其他部分你想怎么定义都行,但要控制内容的格式,对不起,它是由一个名为$content的变量一次输出了整个内容正文。
这就决定了,一般情况下,内容的格式控制只能通过模块来实现,呵呵——不会写程序的同志们可能要晕倒了;其实别的方法也不是没有,就是……通过能写模块的人来实现,哈哈哈,这个是开玩笑哈,另一个方法是修改theme引擎来实现(这个另文阐述,放在这里离题了)。还有非标准的方法,那些个就不介绍了,会误导群众,最后必将导致网站维护、升级异常困难等后果。
下面是一个最简单的页面显示函数,输出我们要的内容:
真够简单,问题是我们该访问那个URL这个页面才会出来呢?下面drupal的url定义钩子hook_menu()隆重登场:
好了,现在访问http://example.com/foo,你可以看到输出的内容了……“Ooops,显示说404 not
found”。这个还是有可能的,因为$may_cache的菜单项都被缓存了,新的路径还没有刷新到缓存里面,此时,要么用devel模块clear
cache,要么浏览一下管理页面中的“菜单”页面就会刷新菜单路径缓存了。现在应该可以了,这样,你在example_foo()的$content里想怎么构建你的页面都行,那个只是html+css的问题了。
顺带讲一下hook_menu()。这个hook的用法蛮灵活,复杂的导航必备,例如通过定义MENU_LOCAL_TASK类型的路径,可以在其他模块产生的页面上的Secondary
Tabs部分嵌入自己的页面。前面的例子里,我们在$items数组里用数组定义了页面的URL相对路径“path”,页面标题“title”,回调函数“callback”和权限控制“access”,整一个意思就是告诉drupal的菜单系统“如果访问foo这个路径,那就调用example_foo这个函数,产生的页面的标题是foo,同时允许任意用户访问(因为access始终是TRUE)”。
下面是一个带参数的url地址定义:
如果用户访问http://example.com/?q=bar/baz,菜单系统会执行example_baz(),如果用户访问http://example.com/?q=bar/baz/1/2,菜单系统会首先查找bar/baz/1/2,如果找不到对应定义会接着找bar/baz/1。如果又找不到,它就会找bar/baz,这样它就会执行example_baz(1,2)。注意路径中的数字部分作为参数传递给了函数,这个实在是非常好用。
如果用户访问http://example.com/?q=bar/baz/52/97,菜单系统找到了匹配,但由于回调函数被省略,因此它最终调用的是example_baz(52,97)。有什么不同呢,此时页面标题不再是“baz”,而是“the
magic numbers”了。
前面所有的路径定义中access都为TRUE,但通常我们都希望对页面内容的访问作一些权限控制。此时我们需要在模块里实现hook_perm()函数:
现在,你在“访问控制”页面就可以分配这两个权限给指定的角色了。同时,你还要在hook_menu里这样定义'access':
user_access()函数会访问$user全局变量和权限设定以确定访问页面的当前用户是否有'access
foo'这个权限,有就返回TRUE,没有返回FALSE。用户ID为1的用户默认拥有任何权限,即user_access(‘任意值’)对他来说都会返回TRUE;权限管理对他完全没用,乖乖。——所以,有特殊要求时,记得还要控制uid=1的用户。
好了,我相信你已经能够用模块定义自己的页面了——虽然看似有些繁琐,但恭喜你已经迈出模块之旅的第一步。
许多时候我们需要为一些数据显示一个自定义格式的页面。熟悉模板的同志们可能曾经失望的发现,模板只能控制除$content之外的那部分页面。在模板里,内容区之外的其他部分你想怎么定义都行,但要控制内容的格式,对不起,它是由一个名为$content的变量一次输出了整个内容正文。
这就决定了,一般情况下,内容的格式控制只能通过模块来实现,呵呵——不会写程序的同志们可能要晕倒了;其实别的方法也不是没有,就是……通过能写模块的人来实现,哈哈哈,这个是开玩笑哈,另一个方法是修改theme引擎来实现(这个另文阐述,放在这里离题了)。还有非标准的方法,那些个就不介绍了,会误导群众,最后必将导致网站维护、升级异常困难等后果。
下面是一个最简单的页面显示函数,输出我们要的内容:
function example_foo() { $content = '<p>The quick brown fox jumps over the lazy dog.</p>'; return $content; }
真够简单,问题是我们该访问那个URL这个页面才会出来呢?下面drupal的url定义钩子hook_menu()隆重登场:
function example_menu($may_cache) { $items = array(); // $may_cache参数用来将菜单项分为两类。 // $may_cache 为 TRUE时返回的菜单项对当前用户在任何时候都可用(并被缓存); // 其他的则是可更改的或只在一定的路径下才被定义的,例如带参数的动态路径。 // 绝大多数模块都会有可缓存的菜单项。 if ($may_cache) { // 这是你必须提供的最基本信息 $items[] = array('path' => 'foo', 'title' => t('foo'), 'callback' => 'example_foo', 'access' => TRUE); } return $items; }
好了,现在访问http://example.com/foo,你可以看到输出的内容了……“Ooops,显示说404 not
found”。这个还是有可能的,因为$may_cache的菜单项都被缓存了,新的路径还没有刷新到缓存里面,此时,要么用devel模块clear
cache,要么浏览一下管理页面中的“菜单”页面就会刷新菜单路径缓存了。现在应该可以了,这样,你在example_foo()的$content里想怎么构建你的页面都行,那个只是html+css的问题了。
顺带讲一下hook_menu()。这个hook的用法蛮灵活,复杂的导航必备,例如通过定义MENU_LOCAL_TASK类型的路径,可以在其他模块产生的页面上的Secondary
Tabs部分嵌入自己的页面。前面的例子里,我们在$items数组里用数组定义了页面的URL相对路径“path”,页面标题“title”,回调函数“callback”和权限控制“access”,整一个意思就是告诉drupal的菜单系统“如果访问foo这个路径,那就调用example_foo这个函数,产生的页面的标题是foo,同时允许任意用户访问(因为access始终是TRUE)”。
下面是一个带参数的url地址定义:
function example_menu($may_cache) { $items = array(); // 通过使用MENU_CALLBACK类型的路径,我们可以为指定路径注册一个 // 回调函数而不出现在菜单项列表里,管理员也不能在菜单管理里禁用这个路径 $items[] = array('path' => 'bar/baz', 'title' => t('baz'), 'callback' => 'example_baz', 'access' => TRUE, 'type' => MENU_CALLBACK); // 下面的菜单项没有注册回调函数,此时,属性就会从父路径继承。 // 例如,这里也会使用父路径的权限控制。不过如果路径上有指定参数的话, // 我们重定义了标题。 // 注意:如果没有'type'属性,此项会在菜单中显示,也就是说'type'不被继承。 $items[] = array('path' => 'bar/baz/52/97', 'title' => t('the magic numbers'), 'type' => MENU_CALLBACK); } return $items; } function example_baz($alice = 0, $bob = 0) { // 永远不要相信URL中传来的值是安全的!一定要记得检查这些值。 if (!is_numeric($alice) || !is_numeric($bob)) { // 如果参数都不是数字,我们将显示一个标准的“你无权访问”的页面 drupal_access_denied(); return; } $list[] = "Alice's number was $alice."; $list[] = "Bob's number was $bob."; $list[] = 'The total was '. ($alice + $bob) .'.'; // 调用theme函数实现输出的格式化,theme_item_list()只是许多theme函数中的一个 $content = theme('item_list', $list); return $content; }
如果用户访问http://example.com/?q=bar/baz,菜单系统会执行example_baz(),如果用户访问http://example.com/?q=bar/baz/1/2,菜单系统会首先查找bar/baz/1/2,如果找不到对应定义会接着找bar/baz/1。如果又找不到,它就会找bar/baz,这样它就会执行example_baz(1,2)。注意路径中的数字部分作为参数传递给了函数,这个实在是非常好用。
如果用户访问http://example.com/?q=bar/baz/52/97,菜单系统找到了匹配,但由于回调函数被省略,因此它最终调用的是example_baz(52,97)。有什么不同呢,此时页面标题不再是“baz”,而是“the
magic numbers”了。
前面所有的路径定义中access都为TRUE,但通常我们都希望对页面内容的访问作一些权限控制。此时我们需要在模块里实现hook_perm()函数:
function example_perm() { return array('access foo', 'access baz'); }
现在,你在“访问控制”页面就可以分配这两个权限给指定的角色了。同时,你还要在hook_menu里这样定义'access':
'access' => user_access('access foo'),
user_access()函数会访问$user全局变量和权限设定以确定访问页面的当前用户是否有'access
foo'这个权限,有就返回TRUE,没有返回FALSE。用户ID为1的用户默认拥有任何权限,即user_access(‘任意值’)对他来说都会返回TRUE;权限管理对他完全没用,乖乖。——所以,有特殊要求时,记得还要控制uid=1的用户。
好了,我相信你已经能够用模块定义自己的页面了——虽然看似有些繁琐,但恭喜你已经迈出模块之旅的第一步。
相关文章推荐
- Drupal&nbsp;模块开发基本教程(一)
- Drupal&nbsp;模块开发基本教程(三)
- Drupal 模块开发基本教程(三)
- Drupal&nbsp;模块开发实例之一:表单的…
- Drupal 模块开发基本教程(二)
- Drupal 模块开发基本教程(一)
- ECshop 二次开发模板教程1
- PhpStorm创建Drupal模块项目开发教程(3)
- 解剖Nginx·模块开发篇(3)ngx_http_hello_world_module 模块的基本函数实现
- PhpStorm创建Drupal模块项目开发教程(5)
- Tribon Vitesse开发教程(一)
- 解剖Nginx·模块开发篇(3)ngx_http_hello_world_module 模块的基本函数实现
- Tribon Vitesse开发教程(二)
- phpcms v9 模块开发基本教程。。。。。新手必看
- Tribon Vitesse开发教程(三)
- ECshop 二次开发模板教程3
- PhpStorm创建Drupal模块项目开发教程(1)
- Drupal8开发教程:模块开发——创建新页面
- Tribon Vitesse开发教程(四)
- Tribon Vitesse开发教程(五)