DPlayer-Typecho/Plugin.php

301 lines
12 KiB
PHP
Raw Normal View History

2017-07-27 08:52:16 +08:00
<?php
if (!defined('__TYPECHO_ROOT_DIR__')) exit;
/**
* DPlayer for typecho
*
* @package DPlayer
* @author Volio
2020-01-12 15:56:02 +08:00
* @version 1.1.0
* @link https://niconiconi.org
2017-07-27 08:52:16 +08:00
*/
class DPlayer_Plugin implements Typecho_Plugin_Interface
{
/**
* 激活插件方法,如果激活失败,直接抛出异常
*
* @access public
* @return void
*/
public static function activate()
{
2020-01-12 15:56:02 +08:00
Typecho_Plugin::factory('Widget_Abstract_Contents')->contentEx = ['DPlayer_Plugin', 'replacePlayer'];
Typecho_Plugin::factory('Widget_Abstract_Contents')->excerptEx = ['DPlayer_Plugin', 'replacePlayer'];
2019-11-22 00:13:37 +08:00
Typecho_Plugin::factory('Widget_Archive')->header = ['DPlayer_Plugin', 'playerHeader'];
Typecho_Plugin::factory('Widget_Archive')->footer = ['DPlayer_Plugin', 'playerFooter'];
Typecho_Plugin::factory('admin/write-post.php')->bottom = ['DPlayer_Plugin', 'addEditorButton'];
Typecho_Plugin::factory('admin/write-page.php')->bottom = ['DPlayer_Plugin', 'addEditorButton'];
2017-07-27 08:52:16 +08:00
}
/**
* 禁用插件方法,如果禁用失败,直接抛出异常
*
* @static
* @access public
* @return void
*/
public static function deactivate()
{
}
/**
* 插入顶部代码
*/
public static function playerHeader()
{
2017-07-27 09:44:41 +08:00
echo <<<EOF
2019-12-09 21:33:39 +08:00
<link rel="stylesheet" type="text/css" href="https://cdn.jsdelivr.net/npm/dplayer/dist/DPlayer.min.css" />
2017-07-27 09:44:41 +08:00
EOF;
2017-07-27 08:52:16 +08:00
}
/**
* 插入底部代码
2019-11-22 00:13:37 +08:00
* @throws Typecho_Exception
2017-07-27 08:52:16 +08:00
*/
public static function playerFooter()
{
$url = Helper::options()->pluginUrl . '/DPlayer';
if (Typecho_Widget::widget('Widget_Options')->plugin('DPlayer')->hls) {
2019-01-13 22:07:00 +08:00
echo "<script type=\"text/javascript\" src=\"$url/plugin/hls.min.js\"></script>\n";
2017-07-27 08:52:16 +08:00
}
if (Typecho_Widget::widget('Widget_Options')->plugin('DPlayer')->flv) {
2019-01-13 22:07:00 +08:00
echo "<script type=\"text/javascript\" src=\"$url/plugin/flv.min.js\"></script>\n";
2017-07-27 08:52:16 +08:00
}
echo <<<EOF
2019-12-09 21:33:39 +08:00
<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/dplayer/dist/DPlayer.min.js"></script>
2020-01-12 15:56:02 +08:00
<script type="text/javascript" src="$url/assets/player.js"></script>
2017-07-27 08:52:16 +08:00
EOF;
}
/**
* 内容标签替换
*
2020-01-12 15:56:02 +08:00
* @param $text
2019-11-22 00:13:37 +08:00
* @param $widget
2020-01-12 15:56:02 +08:00
* @param $last
2017-07-27 08:52:16 +08:00
* @return string
*/
2020-01-12 15:56:02 +08:00
public static function replacePlayer($text, $widget, $last)
2017-07-27 08:52:16 +08:00
{
2020-01-12 15:56:02 +08:00
$text = empty($last) ? $text : $last;
2017-07-27 08:52:16 +08:00
if ($widget instanceof Widget_Archive) {
2019-11-22 00:13:37 +08:00
$pattern = self::get_shortcode_regex(['dplayer']);
2020-01-12 15:56:02 +08:00
$text = preg_replace_callback("/$pattern/", ['DPlayer_Plugin', 'parseCallback'], $text);
2017-07-27 08:52:16 +08:00
}
2020-01-12 15:56:02 +08:00
return $text;
2017-07-27 08:52:16 +08:00
}
/**
* 回调解析
2019-11-22 00:13:37 +08:00
* @param $matches
2017-07-27 08:52:16 +08:00
* @return string
2019-11-22 00:13:37 +08:00
* @throws Typecho_Exception
2017-07-27 08:52:16 +08:00
*/
public static function parseCallback($matches)
{
/*
$mathes array
* 1 - An extra [ to allow for escaping shortcodes with double [[]]
* 2 - The shortcode name
* 3 - The shortcode argument list
* 4 - The self closing /
* 5 - The content of a shortcode when it wraps some content.
* 6 - An extra ] to allow for escaping shortcodes with double [[]]
*/
// allow [[player]] syntax for escaping the tag
if ($matches[1] == '[' && $matches[6] == ']') {
return substr($matches[0], 1, -1);
}
//还原转义后的html
//[dplayer title=&quot;Test Abc&quot; artist=&quot;haha&quot; id=&quot;1234543&quot;/]
2020-01-12 15:56:02 +08:00
$tag = htmlspecialchars_decode($matches[3]);
2017-07-27 08:52:16 +08:00
//[dplayer]标签的属性类型为array
2020-01-12 15:56:02 +08:00
$attrs = self::shortcode_parse_atts($tag);
return DPlayer_Plugin::parsePlayer($attrs);
}
2017-07-27 08:52:16 +08:00
2020-01-12 15:56:02 +08:00
public static function parsePlayer($attrs)
{
2017-07-27 08:52:16 +08:00
//播放器设置
2020-01-12 15:56:02 +08:00
$theme = Typecho_Widget::widget('Widget_Options')->plugin('DPlayer')->theme ?: '#FADFA3';
2017-07-27 08:52:16 +08:00
$api = Typecho_Widget::widget('Widget_Options')->plugin('DPlayer')->api;
2020-01-12 15:56:02 +08:00
//播放器属性
$config = [
2019-01-13 22:07:00 +08:00
'live' => false,
2020-01-12 15:56:02 +08:00
'autoplay' => isset($attrs['autoplay']) && $attrs['autoplay'] == 'true',
'theme' => isset($attrs['theme']) ? $attrs['theme'] : $theme,
'loop' => isset($attrs['loop']) && $attrs['loop'] == 'true',
'screenshot' => isset($attrs['screenshot']) && $attrs['screenshot'] == 'true',
2019-01-13 22:07:00 +08:00
'hotkey' => true,
'preload' => 'metadata',
2020-01-12 15:56:02 +08:00
'lang' => isset($attrs['lang']) ? $attrs['lang'] : 'zh-cn',
'logo' => isset($attrs['logo']) ? $attrs['logo'] : null,
'volume' => isset($attrs['volume']) ? $attrs['volume'] : 0.7,
2019-01-13 22:07:00 +08:00
'mutex' => true,
2020-01-12 15:56:02 +08:00
'video' => [
'url' => isset($attrs['url']) ? $attrs['url'] : null,
'pic' => isset($attrs['pic']) ? $attrs['pic'] : null,
'type' => isset($attrs['type']) ? $attrs['type'] : 'auto',
'thumbnails' => isset($attrs['thumbnails']) ? $attrs['thumbnails'] : null,
],
2019-11-22 00:13:37 +08:00
];
2020-01-12 15:56:02 +08:00
if (isset($attrs['danmu']) && $attrs['danmu'] == 'true') {
$config['danmaku'] = [
'id' => md5(isset($attrs['url']) ? $attrs['url'] : ''),
'api' => $api,
'maximum' => isset($attrs['maximum']) ? $attrs['maximum'] : 1000,
'user' => isset($attrs['user']) ? $attrs['user'] : 'DIYgod',
'bottom' => isset($attrs['bottom']) ? $attrs['bottom'] : '15%',
'unlimited' => true,
];
}
if (isset($attrs['subtitle']) && $attrs['subtitle'] == 'true') {
$config['subtitle'] = [
'url' => isset($attrs['subtitleurl']) ? $attrs['subtitleurl'] : null,
'type' => isset($attrs['subtitletype']) ? $attrs['subtitletype'] : 'webvtt',
'fontSize' => isset($attrs['subtitlefontsize']) ? $attrs['subtitlefontsize'] : '25px',
'bottom' => isset($attrs['subtitlebottom']) ? $attrs['subtitlebottom'] : '10%',
'color' => isset($attrs['subtitlecolor']) ? $attrs['subtitlecolor'] : '#b7daff',
];
}
$json = json_encode($config);
return "<div class=\"dplayer\" data-config='{$json}'></div>";
2017-07-27 08:52:16 +08:00
}
2019-11-22 00:13:37 +08:00
public static function addEditorButton()
2019-01-28 13:19:56 +08:00
{
2020-01-12 15:56:02 +08:00
$dir = Helper::options()->pluginUrl . '/DPlayer/assets/editor.js';
2019-01-28 13:19:56 +08:00
echo "<script type=\"text/javascript\" src=\"{$dir}\"></script>";
}
2017-07-27 08:52:16 +08:00
public static function config(Typecho_Widget_Helper_Form $form)
{
$theme = new Typecho_Widget_Helper_Form_Element_Text(
'theme', null, '#FADFA3',
2020-01-12 15:56:02 +08:00
_t('默认主题颜色'), _t('播放器默认的主题颜色,例如 #372e21、#75c、red、blue该设定会被[dplayer]标签中的theme属性覆盖默认为 #FADFA3'));
2017-07-27 08:52:16 +08:00
$api = new Typecho_Widget_Helper_Form_Element_Text(
2020-01-12 15:56:02 +08:00
'api', null, '',
_t('弹幕服务器地址'), _t('用于保存视频弹幕,例如 https://api.prprpr.me/dplayer/v3/'));
2017-07-27 09:44:41 +08:00
$hls = new Typecho_Widget_Helper_Form_Element_Radio('hls', array('0' => _t('不开启HLS支持'), '1' => _t('开启HLS支持')), '0', _t('HLS支持'), _t("开启后可解析 m3u8 格式视频"));
$flv = new Typecho_Widget_Helper_Form_Element_Radio('flv', array('0' => _t('不开启FLV支持'), '1' => _t('开启FLV支持')), '0', _t('FLV支持'), _t("开启后可解析 flv 格式视频"));
2017-07-27 08:52:16 +08:00
$form->addInput($theme);
$form->addInput($api);
$form->addInput($hls);
$form->addInput($flv);
}
public static function personalConfig(Typecho_Widget_Helper_Form $form)
{
}
/**
* Retrieve all attributes from the shortcodes tag.
*
* The attributes list has the attribute name as the key and the value of the
* attribute as the value in the key/value pair. This allows for easier
* retrieval of the attributes, since all attributes have to be known.
*
* @link https://github.com/WordPress/WordPress/blob/master/wp-includes/shortcodes.php
* @since 2.5.0
*
* @param string $text
* @return array|string List of attribute values.
* Returns empty array if trim( $text ) == '""'.
* Returns empty string if trim( $text ) == ''.
* All other matches are checked for not empty().
*/
private static function shortcode_parse_atts($text)
{
$atts = array();
$pattern = '/([\w-]+)\s*=\s*"([^"]*)"(?:\s|$)|([\w-]+)\s*=\s*\'([^\']*)\'(?:\s|$)|([\w-]+)\s*=\s*([^\s\'"]+)(?:\s|$)|"([^"]*)"(?:\s|$)|(\S+)(?:\s|$)/';
$text = preg_replace("/[\x{00a0}\x{200b}]+/u", " ", $text);
if (preg_match_all($pattern, $text, $match, PREG_SET_ORDER)) {
foreach ($match as $m) {
if (!empty($m[1]))
$atts[strtolower($m[1])] = stripcslashes($m[2]);
elseif (!empty($m[3]))
$atts[strtolower($m[3])] = stripcslashes($m[4]);
elseif (!empty($m[5]))
$atts[strtolower($m[5])] = stripcslashes($m[6]);
elseif (isset($m[7]) && strlen($m[7]))
$atts[] = stripcslashes($m[7]);
elseif (isset($m[8]))
$atts[] = stripcslashes($m[8]);
}
// Reject any unclosed HTML elements
foreach ($atts as &$value) {
if (false !== strpos($value, '<')) {
if (1 !== preg_match('/^[^<]*+(?:<[^>]*+>[^<]*+)*+$/', $value)) {
$value = '';
}
}
}
} else {
$atts = ltrim($text);
}
return $atts;
}
/**
* Retrieve the shortcode regular expression for searching.
*
* The regular expression combines the shortcode tags in the regular expression
* in a regex class.
*
* The regular expression contains 6 different sub matches to help with parsing.
*
* 1 - An extra [ to allow for escaping shortcodes with double [[]]
* 2 - The shortcode name
* 3 - The shortcode argument list
* 4 - The self closing /
* 5 - The content of a shortcode when it wraps some content.
* 6 - An extra ] to allow for escaping shortcodes with double [[]]
*
* @link https://github.com/WordPress/WordPress/blob/master/wp-includes/shortcodes.php
* @since 2.5.0
*
*
* @param array $tagnames List of shortcodes to find. Optional. Defaults to all registered shortcodes.
* @return string The shortcode search regular expression
*/
private static function get_shortcode_regex($tagnames = null)
{
$tagregexp = join('|', array_map('preg_quote', $tagnames));
// WARNING! Do not change this regex without changing do_shortcode_tag() and strip_shortcode_tag()
// Also, see shortcode_unautop() and shortcode.js.
return
'\\[' // Opening bracket
. '(\\[?)' // 1: Optional second opening bracket for escaping shortcodes: [[tag]]
. "($tagregexp)" // 2: Shortcode name
. '(?![\\w-])' // Not followed by word character or hyphen
. '(' // 3: Unroll the loop: Inside the opening shortcode tag
. '[^\\]\\/]*' // Not a closing bracket or forward slash
. '(?:'
. '\\/(?!\\])' // A forward slash not followed by a closing bracket
. '[^\\]\\/]*' // Not a closing bracket or forward slash
. ')*?'
. ')'
. '(?:'
. '(\\/)' // 4: Self closing tag ...
. '\\]' // ... and closing bracket
. '|'
. '\\]' // Closing bracket
. '(?:'
. '(' // 5: Unroll the loop: Optionally, anything between the opening and closing shortcode tags
. '[^\\[]*+' // Not an opening bracket
. '(?:'
. '\\[(?!\\/\\2\\])' // An opening bracket not followed by the closing shortcode tag
. '[^\\[]*+' // Not an opening bracket
. ')*+'
. ')'
. '\\[\\/\\2\\]' // Closing shortcode tag
. ')?'
. ')'
. '(\\]?)'; // 6: Optional second closing brocket for escaping shortcodes: [[tag]]
}
2019-11-21 23:36:02 +08:00
}