您的位置:首页 > 其它

Drupal 模块开发基本教程(二)

2012-11-29 14:19 232 查看
第二部分:模块的自定义页面显示方法

许多时候我们需要为一些数据显示一个自定义格式的页面。熟悉模板的同志们可能曾经失望的发现,模板只能控制除$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的用户。

好了,我相信你已经能够用模块定义自己的页面了——虽然看似有些繁琐,但恭喜你已经迈出模块之旅的第一步。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: