221 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			221 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
| <?php
 | |
| // +----------------------------------------------------------------------
 | |
| // | ThinkPHP [ WE CAN DO IT JUST THINK ]
 | |
| // +----------------------------------------------------------------------
 | |
| // | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
 | |
| // +----------------------------------------------------------------------
 | |
| // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
 | |
| // +----------------------------------------------------------------------
 | |
| // | Author: liu21st <liu21st@gmail.com>
 | |
| // +----------------------------------------------------------------------
 | |
| 
 | |
| namespace think;
 | |
| 
 | |
| class Hook
 | |
| {
 | |
|     /**
 | |
|      * 钩子行为定义
 | |
|      * @var array
 | |
|      */
 | |
|     private $tags = [];
 | |
| 
 | |
|     /**
 | |
|      * 绑定行为列表
 | |
|      * @var array
 | |
|      */
 | |
|     protected $bind = [];
 | |
| 
 | |
|     /**
 | |
|      * 入口方法名称
 | |
|      * @var string
 | |
|      */
 | |
|     private static $portal = 'run';
 | |
| 
 | |
|     /**
 | |
|      * 应用对象
 | |
|      * @var App
 | |
|      */
 | |
|     protected $app;
 | |
| 
 | |
|     public function __construct(App $app)
 | |
|     {
 | |
|         $this->app = $app;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 指定入口方法名称
 | |
|      * @access public
 | |
|      * @param  string  $name     方法名
 | |
|      * @return $this
 | |
|      */
 | |
|     public function portal($name)
 | |
|     {
 | |
|         self::$portal = $name;
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 指定行为标识 便于调用
 | |
|      * @access public
 | |
|      * @param  string|array  $name     行为标识
 | |
|      * @param  mixed         $behavior 行为
 | |
|      * @return $this
 | |
|      */
 | |
|     public function alias($name, $behavior = null)
 | |
|     {
 | |
|         if (is_array($name)) {
 | |
|             $this->bind = array_merge($this->bind, $name);
 | |
|         } else {
 | |
|             $this->bind[$name] = $behavior;
 | |
|         }
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 动态添加行为扩展到某个标签
 | |
|      * @access public
 | |
|      * @param  string    $tag 标签名称
 | |
|      * @param  mixed     $behavior 行为名称
 | |
|      * @param  bool      $first 是否放到开头执行
 | |
|      * @return void
 | |
|      */
 | |
|     public function add($tag, $behavior, $first = false)
 | |
|     {
 | |
|         isset($this->tags[$tag]) || $this->tags[$tag] = [];
 | |
| 
 | |
|         if (is_array($behavior) && !is_callable($behavior)) {
 | |
|             if (!array_key_exists('_overlay', $behavior)) {
 | |
|                 $this->tags[$tag] = array_merge($this->tags[$tag], $behavior);
 | |
|             } else {
 | |
|                 unset($behavior['_overlay']);
 | |
|                 $this->tags[$tag] = $behavior;
 | |
|             }
 | |
|         } elseif ($first) {
 | |
|             array_unshift($this->tags[$tag], $behavior);
 | |
|         } else {
 | |
|             $this->tags[$tag][] = $behavior;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 批量导入插件
 | |
|      * @access public
 | |
|      * @param  array     $tags 插件信息
 | |
|      * @param  bool      $recursive 是否递归合并
 | |
|      * @return void
 | |
|      */
 | |
|     public function import(array $tags, $recursive = true)
 | |
|     {
 | |
|         if ($recursive) {
 | |
|             foreach ($tags as $tag => $behavior) {
 | |
|                 $this->add($tag, $behavior);
 | |
|             }
 | |
|         } else {
 | |
|             $this->tags = $tags + $this->tags;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 获取插件信息
 | |
|      * @access public
 | |
|      * @param  string $tag 插件位置 留空获取全部
 | |
|      * @return array
 | |
|      */
 | |
|     public function get($tag = '')
 | |
|     {
 | |
|         if (empty($tag)) {
 | |
|             //获取全部的插件信息
 | |
|             return $this->tags;
 | |
|         }
 | |
| 
 | |
|         return array_key_exists($tag, $this->tags) ? $this->tags[$tag] : [];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 监听标签的行为
 | |
|      * @access public
 | |
|      * @param  string $tag    标签名称
 | |
|      * @param  mixed  $params 传入参数
 | |
|      * @param  bool   $once   只获取一个有效返回值
 | |
|      * @return mixed
 | |
|      */
 | |
|     public function listen($tag, $params = null, $once = false)
 | |
|     {
 | |
|         $results = [];
 | |
|         $tags    = $this->get($tag);
 | |
| 
 | |
|         foreach ($tags as $key => $name) {
 | |
|             $results[$key] = $this->execTag($name, $tag, $params);
 | |
| 
 | |
|             if (false === $results[$key] || (!is_null($results[$key]) && $once)) {
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $once ? end($results) : $results;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 执行行为
 | |
|      * @access public
 | |
|      * @param  mixed     $class  行为
 | |
|      * @param  mixed     $params 参数
 | |
|      * @return mixed
 | |
|      */
 | |
|     public function exec($class, $params = null)
 | |
|     {
 | |
|         if ($class instanceof \Closure || is_array($class)) {
 | |
|             $method = $class;
 | |
|         } else {
 | |
|             if (isset($this->bind[$class])) {
 | |
|                 $class = $this->bind[$class];
 | |
|             }
 | |
|             $method = [$class, self::$portal];
 | |
|         }
 | |
| 
 | |
|         return $this->app->invoke($method, [$params]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 执行某个标签的行为
 | |
|      * @access protected
 | |
|      * @param  mixed     $class  要执行的行为
 | |
|      * @param  string    $tag    方法名(标签名)
 | |
|      * @param  mixed     $params 参数
 | |
|      * @return mixed
 | |
|      */
 | |
|     protected function execTag($class, $tag = '', $params = null)
 | |
|     {
 | |
|         $method = Loader::parseName($tag, 1, false);
 | |
| 
 | |
|         if ($class instanceof \Closure) {
 | |
|             $call  = $class;
 | |
|             $class = 'Closure';
 | |
|         } elseif (is_array($class) || strpos($class, '::')) {
 | |
|             $call = $class;
 | |
|         } else {
 | |
|             $obj = Container::get($class);
 | |
| 
 | |
|             if (!is_callable([$obj, $method])) {
 | |
|                 $method = self::$portal;
 | |
|             }
 | |
| 
 | |
|             $call  = [$class, $method];
 | |
|             $class = $class . '->' . $method;
 | |
|         }
 | |
| 
 | |
|         $result = $this->app->invoke($call, [$params]);
 | |
| 
 | |
|         return $result;
 | |
|     }
 | |
| 
 | |
|     public function __debugInfo()
 | |
|     {
 | |
|         $data = get_object_vars($this);
 | |
|         unset($data['app']);
 | |
| 
 | |
|         return $data;
 | |
|     }
 | |
| }
 |