什么是OneThink oneThink后台添加插件步骤

什么是OneThink oneThink后台添加插件步骤,第1张

OneThink特性介绍:

1. 基于ThinkPHP最新版本Thinkphp3.2。

2. 模块化:全新的架构和模块化的开发机制,便于灵活扩展和二次开发。

3. 文档模型/分类体系:通过和文档模型绑定,以及不同的文档类型,不同分类可以实现差异化的功能,轻松实现诸如资讯、下载、讨论和图片等功能。

4. 开源免费:OneThink遵循Apache2开源协议,免费提供使用。

5. 用户行为:支持自定义用户行为,可以对单个用户或者群体用户的行为进行记录及分享,为您的运营决策提供有效参考数据。

6. 云端部署:通过驱动的方式可以轻松支持平台的部署,让您的网站无缝迁移,内置已经支持SAE。

7. 云服务支持:即将启动支持云存储、云安全、云过滤和云统计等服务,更多贴心的服务让您的网站更安心。

8. 安全稳健:提供稳健的安全策略,包括备份恢复、容错、防止恶意攻击登录,网页防篡改等多项安全管理功能,保证系统安全,可靠、稳定的运行。

9. 应用仓库:官方应用仓库拥有大量来自第三方插件和应用模块、模板主题,有众多来自开源社区的贡献,让您的网站“One”美无缺。

oneThink后台添加插件步骤:

版本:V1.1.141212 (注:v1.1也有很多版本,一不小心就下到V1.1.140202 去了,还有其他版本,建议去代码托管平台下载最新版本)

我也不偷懒,把每一步步骤都记录下来。

一、进入后台,创建插件

这里的钩子我新建了一个indexFooter,因为我只需在前台首页底部显示友情链接即可。我们把上面所有要勾的地方都勾上,至于有什么区别,大家可以自己建几个例子区分一下,生成的文件是否一样。OK!到这里

我们的友情链接插件就创建好了!点击“确定”。(这里的自定义模板什么的,通通不填,我会在下篇文章演示添加自定义模板的效果)

二、点击“安装” 即可,找到我们刚安装好的Links插件,点击“设置”,你会看到它有个默认的“是否开启随机”的选项,这里我们不管它,因为我们用不上,等下要删掉的。安装后,我们可以在左侧导航“已安装插件后台”看到我们新建的“友情链接”

三、当我们点击左侧导航的“友情链接”,你会发现报错,大概就是说的某个表不存在。是的,我们刚才只是建了插件,如果涉及到数据存储到数据库,

还需要建表。这里不直接去数据库里建,因为这样做,是很不人性化的。那我们就找到安装插件的函数,在安装插件的时候建立数据库,这样就好了。首先系统的插件全部存放在

根目录/Addons/

文件夹下面,打开此文件夹,我们看到有个Links文件夹,这就是我们刚创建的插件,一个插件对应一个文件夹。打开Links文件夹,里面有2个文件和2个文件夹。

四、其实现在oneThink做的越来越简洁了,不懂PHP的人照样创建插件,之后你就会发现。当然,如果你有自己的想法,不想局限于官方的限制,那还是要把php学好的。

五、打开插件入口文件:LinksAddon.class.php 里面有个类LinksAddon,先来分析一下这个文件吧

我这里把$admin_list 数组的model 值改成links了,为了与插件对应。接下来我们在install方法里添加新建数据库的语句,这样我们在安装插件的时候,就会新建数据库了,我的代码如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

public function install(){//安装插件的方法

//1、添加数据表

$model = D()

$db_prefix = C('DB_PREFIX')

$table_name = "{$db_prefix}links"

$sql=<<<SQL

CREATE TABLE IF NOT EXISTS `$table_name` (

`id` int(11) unsigned NOT NULL AUTO_INCREMENT COMMENT '主键',

`title` varchar(80) NOT NULL DEFAULT '' COMMENT '站点名称',

`link` varchar(140) NOT NULL DEFAULT '' COMMENT '链接地址',

`summary` varchar(255) NOT NULL DEFAULT '' COMMENT '站点描述',

`mailto` varchar(100) NOT NULL DEFAULT '' COMMENT '站长联系方式',

`sort` int(3) unsigned NOT NULL DEFAULT 0 COMMENT '优先级',

`nofollow` tinyint(1) unsigned NOT NULL DEFAULT 0 COMMENT '是否追踪',

`type` tinyint(3) unsigned NOT NULL DEFAULT 1 COMMENT '类型分组',

`cover_id` int(11) unsigned NOT NULL DEFAULT 0 COMMENT '封面图片',

`status` tinyint(2) NOT NULL DEFAULT 1 COMMENT '状态(0:禁用,1:正常)',

`create_time` int(11) unsigned NOT NULL DEFAULT 0 COMMENT '添加时间',

PRIMARY KEY (`id`)

) ENGINE=MyISAM AUTO_INCREMENT=4 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='友情连接表'

SQL

$model ->execute($sql)//执行sql语句

//2、返回true,表示插件安装成功

return true

}

我这里省略了很多细节判断,大家自己完善。

六、既然在安装插件的时候,新建了表,我们在卸载的插件的时候就要把表给删除,不然下次安装该插件的时候就会出问题。所以我们uninstall 方法代码如下:

?

1

2

3

4

5

6

7

8

9

public function uninstall(){ //卸载插件的方法

$model = D()

$db_prefix = C('DB_PREFIX')

$table_name = "{$db_prefix}links"

$sql="DROP TABLE IF EXISTS `".$table_name."`"

$model ->execute($sql)//执行sql语句

return true

}

好了,到这里就差不多了,保存一下LinksAddon.class.php 文件,应该可以正常显示了,我们来看看。进入插件列表,先把Links插件卸载,然后重新安装。点击左侧菜单“友情链接”,可以看到

之所以能正常显示这个列表,是因为系统有默认的模板,在\Application\Admin\View\Addons

文件夹里,有兴趣的同学可以研究一下这几个模板文件,其中这个列表的模板就是adminlist.html,那么我们要把封面、书名、描述等等这些字眼改掉,要去模板里改吗?细心的同学估计注意到了,在LinksAddon.class.php

文件 的$admin_list 数组里配置的,其他的看后面的注释就明白,这里详细说一下list_grid

关联的数组。我们刚才新建的links数据表有id、title、link等字段,你想在这个列表显示什么字段,都可以添加。我这里代码如下:

?

1

2

3

4

5

6

7

'list_grid'=>array( //这里定义的是除了id序号外的表格里字段显示的表头名和模型一样支持函数和链接

'title:网站名称',

'link:链接',

'summary:描述',

'create_time|time_format:添加时间', //time_format 是一个函数,把时间格式化,其他地方想使用什么函数也可以按照这种格式书写

'id: *** 作:[EDIT]|编辑,[DELETE]|删除'

),

保存,刷新后台友情链接列表

我们点击“新增” 来增加一个友情链接吧,你会发现,只有一个书名字段。我们打开Model/LinksModel.class.php 文件,我这里分别解释一下这两个自带的数组,具体看下面代码里的注释

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

class LinksModel extends Model{

public $model = array(

'title'=>'',//新增[title]、编辑[title]、删除[title]的提示

'template_add'=>'',//自定义新增模板自定义html edit.html 会读取插件根目录的模板

'template_edit'=>'',//自定义编辑模板html

'search_key'=>'',// 搜索的字段名,默认是title

'extend'=>1, //在后台列表是否显示 “增加”、“删除” 按钮,0-不显示 1-显示

)

public $_fields = array(

'id'=>array(

'name'=>'id',//字段名,与数据库的字段对应

'title'=>'ID',//显示标题

'type'=>'num',//字段类型:num、string、textarea、datetime、bool、select、radio、checkbox、editor、picture(封面)、file(附件)、

'remark'=>'',// 备注,相当于配置里的tip

'is_show'=>3,// 1-始终显示 2-新增显示 3-编辑显示 0-不显示

'value'=>0,//默认值

),

//下面演示一下 select字段怎么显示 radio、checkbox同理

'type'=>array(

'name'=>'type',

'title'=>'类型',

'type'=>'select',

'remark'=>'请选择所属类型',

'is_show'=>1,

 'extra'=>'0:友情链接,1:合作站点',

'value'=>0,

'is_must'=>1,

),

)

}

ok,我最后的效果是这样的:

添加一条数据看看吧:

这里要显示具体类型、显示图片等,需要自定义adminlist.html模板了。关于自定义模板,我们下一篇文章再说。关于钩子,其实就是写一个函数从数据库读取数据,然后在前台需要的地方调用钩子就行。如果需要模板,则在钩子函数里解析模板。钩子调用格式一般:

{:hook("钩子名称"),"[参数]"} 没参数就不写。直接写成这样{:hook("钩子名称")}

到此为止就是用系统默认的模板,一步一步的建立自己的插件,是不是很简单,就像填空题一样,只要按照它的规则填空,就ok了。

钩子就像是一个挂载点,挂到函数上。当函数执行过程中遇到这个挂载点,这个钩子 就会将一块代码拉出来,执行这个代码。钩子函数相对于直接在函数中调用另外一个函数来说,更加安全方便。当我们需要修改扩展功能时,我们无需修改函数B中的钩子,只需要修改钩子里面的代码块即可,而如果直接修改函数A,则会对函数B所在类进行频繁修改。违背了封闭原则。另一点,利用钩子对后期的维护和功能扩展更加方便。

thinkphp里面,钩子就是进行行为扩展。下面是网上一些对行为扩展的解释:

行为

行为(Behavior)是一个比较抽象的概念,你可以想象成在应用执行过程中的一个动作或者处理,在框架的执行流程中,各个位置都可以有行为产生,例如路由检测是一个行为,静态缓存是一个行为,用户权限检测也是行为,大到业务逻辑,小到浏览器检测、多语言检测等等都可以当做是一个行为,甚至说你希望给你的网站用户的第一次访问d出Hello,world!这些都可以看成是一种行为,行为的存在让你无需改动框架和应用,而在外围通过扩展或者配置来改变或者增加一些功能。

而不同的行为之间也具有位置共同性,比如,有些行为的作用位置都是在应用执行前,有些行为都是在模板输出之后,我们把这些行为发生作用的位置称之为标签(位)(tag),当应用程序运行到这个标签的时候,就会被拦截下来,统一执行相关的行为

ThinkPHP3.2 扩展--钩子,HOOK

之前写到TP3.1的行为扩展是tag()在TP3.2中引入了另一种说法—:钩子。

我们来看一下TP3.2中的钩子这个东西:

一:文件流程:

1:/index.php ->require './ThinkPHP/ThinkPHP.php'

2:/ThinkPHP/ThinkPHP.php—->require CORE_PATH.'Think'.EXTThink\Think::start()

3:/ThinkPHP/Library/Think/Think.class.php—–>App::run()

4:/ThinkPHP/Library/Think/App.class.php 。到这里基本流程就走完了,(这里不说细节);

二:代码:

1:看一下 App::run()方法:

// 应用初始化标签

Hook::listen('app_init')

App::init()

// 应用开始标签

Hook::listen('app_begin')

// Session初始化

if(!IS_CLI){

session(C('SESSION_OPTIONS'))

}

// 记录应用初始化时间

G('initTime')

App::exec()

// 应用结束标签

Hook::listen('app_end')

return

其中的Hook::listen(”)就是用来执行钩子的,我们可以在app_init这个安插的位置用来获取应用中安装的插件。

看一下Hook::listen();

/**

* 监听标签的插件

* @param string $tag 标签名称

* @param mixed $params 传入参数

* @return void

*/

static public function listen($tag, &$params=NULL) {

if(isset(self::$tags[$tag])) {

if(APP_DEBUG) {

G($tag.'Start')

trace('[ '.$tag.' ] --START--','','INFO')

}

foreach (self::$tags[$tag] as $name) {

APP_DEBUG &&G($name.'_start')

$result = self::exec($name, $tag,$params)

if(APP_DEBUG){

G($name.'_end')

trace('Run '.$name.' [ RunTime:'.G($name.'_start',$name.'_end',6).'s ]','','INFO')

}

if(false === $result) {

// 如果返回false 则中断插件执行

return

}

}

if(APP_DEBUG) { // 记录行为的执行日志

trace('[ '.$tag.' ] --END-- [ RunTime:'.G($tag.'Start',$tag.'End',6).'s ]','','INFO')

}

}

return

}

其中关键是:self::exec($name, $tag,$params)看一下exec的代码:

/**

* 执行某个插件

* @param string $name 插件名称

* @param string $tag 方法名(标签名)

* @param Mixed $params 传入的参数

* @return void

*/

static public function exec($name, $tag,&$params=NULL) {

if(false === strpos($name,'\\')) {

// 插件(多个入口)

$class = "Addons\\{$name}\\{$name}Addon"

}else{

// 行为扩展(只有一个run入口方法)

$class = $name.'Behavior'

$tag= 'run'

}

$addon = new $class()

return $addon->$tag($params)

}

最后还不是 new $class()进而return $addon->$tag($params)又转到了具体钩子的代码方法。其实就是我们原本的调用class的方法,只不过经过别人的高度封装了。

三:那么问题来了,这个钩子有什么用呢? 怎么用?

这里以OneThink 的{:hook('AdminIndex')}为例,看一些别人是怎么用的。

在系统初始化到 Hook::listen('app_init')时,

把app_init的标签位扩展了,在tags.php的配置文件中有这么个东西:用于初始化插件(或者说是获取系统中安装的插件)

<?php

return array(

'app_init'=>array('Common\Behavior\InitHook')

)

一看就明白,无非就是读取持久化的信息,放到缓存或是其他的方式

// 行为扩展的执行入口必须是run

public function run(&$content){

if(isset($_GET['m']) &&$_GET['m'] === 'Install') return

$data = S('hooks')

if(!$data){

$hooks = M('Hooks')->getField('name,addons')

foreach ($hooks as $key =>$value) {

if($value){

$map['status'] = 1

$names = explode(',',$value)

$map['name']= array('IN',$names)

$data = M('Addons')->where($map)->getField('id,name')

if($data){

$addons = array_intersect($names, $data)

Hook::add($key,$addons)

}

}

}

S('hooks',Hook::get())

}else{

Hook::import($data,false)

}

}

当在程序执行到{:hook(‘AdminIndex’)}时—>调用的是Hook::listen(‘AdminIndex’)

AdminIndex这个挂载点包含了三个插件:分别是:SiteStat, SystemInfo,DevTeam。

用一个循环来分别按顺序执行.

总结:钩子其实就是起到一个挂载点的作用,这个钩子挂在哪里,就可以在哪里执行,内容或功能就是挂载插件或类库的具体实现。这样实现的代码就有很大的灵活性,挂载点不变,挂的东西变量,功能也就相应的变化,是不是很灵活强大呀。


欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/bake/11732900.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-18
下一篇 2023-05-18

发表评论

登录后才能评论

评论列表(0条)

保存