网站内容图片高度移除js方法

 

网站内容图片高度移除js方法。

<div id="delheight">

<img style="width:100%;height:99px;" src="/yourimage.png" />

</div>

<script type="text/javascript"> 
window.onload =function() 
{ 
var e=document.getElementById("delheight"); 
var str1 = e.innerHTML; 
str1 = str1.replace(/(\bheight\s*=\s*(?:'[^']*'|"[^"]*"))/ig, "");
str1 = str1.replace(/[ \t]*height[ \t]*=[ \t]*(?:'[^']*'|"[^"]*")/ig, "");
e.innerHTML = str1; 
} 
</script>

 

 

阿里云邮件推送SMTP服务地址

来源:https://help.aliyun.com/zh/direct-mail/smtp-endpoints

不同站点的SMTP服务地址。

华东 1(杭州): smtpdm.aliyun.com
新加坡(新):smtpdm-ap-southeast-1.aliyuncs.com
美国(新):smtpdm-us-east-1.aliyuncs.com
德国:smtpdm-eu-central-1.aliyuncs.com
SMTP端口号:25,80,465(SSL加密)
注意:ECS 基于安全考虑,目前已禁用 25 端口。如果您的发送程序部署在阿里云 ECS 上,建议您不勾选 SSL 时,使用 80 端口;勾选 SSL 时,使用 465 端口。云虚拟主机服务器同上。此外,我们支持通过smtp命令“STARTTLS”在端口25或80上开启显式SSL/TLS功能。
请根据控制台选择的服务区域选择对应的服务器地址

如何选择发信区域

1.前提:不同的地域决定了数据落地的地域。如果涉及境外客户,请先咨询您公司的法务,确定数据在哪个区域才合规。
2.如果您的发信程序部署在境内,建议您选择华东1区域。
3.如果您的发信程序部署在境外,且收信地址主要在境外,建议您使用新加坡、美国等集群。
4.如果您的发信程序部署在境外,且收信地址境内居多:
a)如果您的发信程序使用SMTP发信,且部署在美国或者新加坡,建议您使用华东1区域,会自动通过美国或者新加坡加速节点路由到华东1节点。
b)其他情况,建议您使用新加坡区域。
 

wordpress文章内容生成图片的方法

请自行下载字体AlibabaPuHuiTi-3-45-Light.otf放在主题根目录下。

在主题函数文件functions.php中添加

function ii_get_content2image($imgwidth,$fontsize,$type = '0'){
ini_set('memory_limit', '512M');
$tid = get_the_ID();
$text = get_post_field('post_content', $tid); // 获取文章内容

// 设置文字内容
$textContent = preg_replace("/(<br\s*\/?>)/i", "\n", strip_tags($text,'<p> <br> <img>'));
$textContent = preg_replace('/[\x{1F600}-\x{1F64F}\x{1F300}-\x{1F5FF}\x{1F680}-\x{1F6FF}\x{2600}-\x{26FF}\x{2700}-\x{27BF}\x{1F1E0}-\x{1F1FF}\x{1F900}-\x{1F9FF}\x{1FA00}-\x{1FA6F}\x{1FA70}-\x{1FAFF}\x{1FBC0}-\x{1FBCF}\x{1FBD0}-\x{1FBD0}\x{1FBE0}-\x{1FBE0}\x{1FBF0}-\x{1FBF0}\x{1FC00}-\x{1FC00}\x{1FC70}-\x{1FC70}\x{1FD00}-\x{1FD3E}\x{1FD40}-\x{1FD40}\x{1FD50}-\x{1FD8F}\x{1FD92}-\x{1FD95}\x{1FD97}-\x{1FD97}\x{1FD99}-\x{1FDBF}\x{1FDC0}-\x{1FDCF}\x{1FDD0}-\x{1FDDF}\x{1FDE0}-\x{1FDFD}\x{1FDFE}-\x{1FDFE}\x{1FE00}-\x{1FE0F}\x{1FE10}-\x{1FE1F}\x{1FE20}-\x{1FE2F}\x{1FE30}-\x{1FE4F}\x{1FE50}-\x{1FE50}\x{1FE70}-\x{1FEFF}\x{1FF00}-\x{1FF00}\x{1FF01}-\x{1FF02}\x{1FF03}-\x{1FF03}\x{1FF04}-\x{1FF04}\x{1FF05}-\x{1FF05}\x{1FF06}-\x{1FF06}\x{1FF07}-\x{1FF07}\x{1FF08}-\x{1FF0A}\x{1FF0B}-\x{1FF0B}\x{1FF0C}-\x{1FF0E}\x{1FF0F}-\x{1FF0F}\x{1FF10}-\x{1FF10}\x{1FF11}-\x{1FF11}\x{1FF12}-\x{1FF12}\x{1FF13}-\x{1FF13}\x{1FF14}-\x{1FF14}\x{1FF15}-\x{1FF15}\x{1FF16}-\x{1FF16}\x{1FF17}-\x{1FF17}\x{1FF18}-\x{1FF18}\x{1FF19}-\x{1FF19}\x{1FF1A}-\x{1FF20}\x{1FF21}-\x{1FF21}\x{1FF22}-\x{1FF22}\x{1FF23}-\x{1FF23}\x{1FF24}-\x{1FF24}\x{1FF25}-\x{1FF25}\x{1FF26}-\x{1FF26}\x{1FF27}-\x{1FF27}\x{1FF28}-\x{1FF28}\x{1FF29}-\x{1FF29}\x{1FF2A}-\x{1FF2A}\x{1FF2B}-\x{1FF2B}\x{1FF2C}-\x{1FF2C}\x{1FF2D}-\x{1FF2D}\x{1FF2E}-\x{1FF2F}\x{1FF30}-\x{1FF30}\x{1FF31}-\x{1FF31}\x{1FF32}-\x{1FF32}\x{1FF33}-\x{1FF33}\x{1FF34}-\x{1FF34}\x{1FF35}-\x{1FF35}\x{1FF36}-\x{1FF36}\x{1FF37}-\x{1FF37}\x{1FF38}-\x{1FF38}\x{1FF39}-\x{1FF39}\x{1FF3A}-\x{1FF3A}\x{1FF3B}-\x{1FF3B}\x{1FF3C}-\x{1FF3C}\x{1FF3D}-\x{1FF3D}\x{1FF3E}-\x{1FF3E}\x{1FF3F}-\x{1FF3F}\x{1FF40}-\x{1FF40}\x{1FF41}-\x{1FF41}\x{1FF42}-\x{1FF42}\x{1FF43}-\x{1FF43}\x{1FF44}-\x{1FF44}\x{1FF45}-\x{1FF45}\x{1FF46}-\x{1FF46}\x{1FF47}-\x{1FF47}\x{1FF48}-\x{1FF48}\x{1FF49}-\x{1FF49}\x{1FF4A}-\x{1FF4A}\x{1FF4B}-\x{1FF4B}\x{1FF4C}-\x{1FF4C}\x{1FF4D}-\x{1FF4D}\x{1FF4E}-\x{1FF4F}\x{1FF50}-\x{1FF50}\x{1FF51}-\x{1FF51}\x{1FF52}-\x{1FF52}\x{1FF53}-\x{1FF53}\x{1FF54}-\x{1FF54}\x{1FF55}-\x{1FF55}\x{1FF56}-\x{1FF56}\x{1FF57}-\x{1FF57}\x{1FF58}-\x{1FF58}\x{1FF59}-\x{1FF59}\x{1FF5A}-\x{1FF5A}\x{1FF5B}-\x{1FF5B}\x{1FF5C}-\x{1FF5C}\x{1FF5D}-\x{1FF5D}\x{1FF5E}-\x{1FF5E}\x{1FF5F}-\x{1FF5F}\x{1FF60}-\x{1FF60}\x{1FF61}-\x{1FF61}]/u', '', $textContent);
$textContent = preg_replace('/<p(.*?)>(.*?)<\/p>/s', "$2\n", $textContent);
$textContent = preg_replace("/\s*[\r\n]+|\s*<br\s*\/?>\s*/i", "\n", $textContent);//多个换行替换成1个换行
$text = html_entity_decode($textContent);
// 设置字体文件路径
$font_file = get_template_directory() . '/AlibabaPuHuiTi-3-45-Light.otf'; // 确保这个路径是正确的
$max_width = $imgwidth; // 图片的最大宽度
$font_size = $fontsize; // 文字大小
$image_height = 10; // 初始化图片高度
$line_height = $font_size*2; // 行间距,这里简单加了一些额外空间
$outimg = ABSPATH . 'conimg/';
if (!(is_dir($outimg))) @mkdir($outimg, 0775);
if(!empty($tid)) $outimg .= md5($tid.'_'.$imgwidth.'_'.$fontsize.'_'.$type).'.webp';
else $outimg .= md5(strtotime(time()),1).'.webp';
if(file_exists($outimg)){
$outimg = site_url().'/conimg/'.basename($outimg);
$outimg = str_replace(ABSPATH,site_url(),$outimg);
return $outimg;
}

// 检查字体文件是否存在
if (!file_exists($font_file)) {
die('字体文件不存在');
}
// 创建一个真彩色图像(确保GD库已启用)
$image = imagecreatetruecolor($max_width, $image_height);
$white = imagecolorallocate($image, 255, 255, 255);
$black = imagecolorallocate($image, 0, 0, 0);
imagefill($image, 0, 0, $white);
$parts = preg_split('/(<img[^>]+>)/', $text, -1, PREG_SPLIT_DELIM_CAPTURE);
$isText = true; // 标记当前处理的是文本还是图片
// 精确版的文本分割以适应宽度
$lines = [];
$images = []; // 存储图片的服务器路径和尺寸信息
$current_line_width = 0;
$lines[] = '';//添加空行
foreach ($parts as $part) {
if ($isText) {
// 对文本进行必要的处理,计算内容,行数,高度
foreach (explode("\n", wordwrap($part, $max_width, "\n", true)) as $article) {
preg_match_all("/./u", $article, $words);
foreach ($words[0] as $word) {
if(trim($word) == "") continue;
$word_box = imagettfbbox($font_size, 0, $font_file, $word);
$word_width = $word_box[2] - $word_box[0];
if(preg_match('/[\x{3002}\x{ff01}\x{2018}\x{2019}\x{201c}\x{201d}\x{fe50}\x{fe51}\x{fe52}\x{ff08}\x{ff09}\x{ff0c}\x{ff1a}\x{ff1b}\x{ff1f}\x{ff0e}\x{2026}\x{2014}\x{fe41}\x{fe42}\x{fe43}\x{fe44}\x{ff5e}\x{301d}\x{3001}]/u', $word)) $word_width = $word_width*2;
// 如果加上新单词后超过最大宽度,则开始新的一行
if ($current_line_width + $word_width > $max_width) {
$lines[] = implode('', $current_line);
$current_line = [$word];
$current_line_width = $word_width;
} else {
$current_line[] = $word;
$current_line_width += $word_width + ($font_size /4); // 简单添加字间距
}
}
// 添加最后一行
if (!empty($current_line)) {
$lines[] = implode('', $current_line);
}
$current_line = [];
$current_line_width = 0;
$lines[] = '';//添加空行,隔开段落
}
} else {
// 对图片标签进行处理
// 处理图片标签并记录图片信息
preg_match_all('/<img[^>]*src=["\'](.*?)["\']/i', $part, $matches);
foreach ($matches[1] as $imageSrc) {
// 假设$imageSrc已经是服务器相对或绝对路径
if ((strpos($imageSrc, 'http') !== 0 || strpos($imageSrc, 'https') !== 0) && strpos($imageSrc, '//') !== 0) {
$imageSrc = site_url() . '/' .$imageSrc;
}
$imageSrc = str_replace(site_url(),ABSPATH,$imageSrc);
$imageSrc = str_replace('//','/',$imageSrc);
if (!file_exists($imageSrc)) continue;
list($width, $height) = @getimagesize($imageSrc);
if($width >= $max_width){
$newWidth = $max_width * 0.85;
$newHeight = $height * ($newWidth/$width);
}else{
$newWidth = $width;
$newHeight = $height;
}
$images[] = array(
'src' => $imageSrc,
'x' => ($max_width - $newWidth) / 2,
'y' => count($lines) * $line_height,
'w' => $width,
'h' => $height,
'nw' => $newWidth,
'nh' => $newHeight
);
}
if(count($images) > 0){
$imagesH = ceil($newHeight/$line_height);
for($i=0;$i<=$imagesH;$i++){
$lines[] = '';//添加空行
}
}
}
$isText = !$isText; // 切换到处理下一种类型(文本或图片)
}

// 计算总高度并重新创建图像以适应所有文字
$image_height = count($lines) * $line_height;
if($image_height > 10000) {
$image_height = '10000';//图片超高,仅返回图片前1万像素高度
$lines_len = ceil($image_height/$line_height) - 1;
$lines[$lines_len-3] = '';
$lines[$lines_len-2] = '图片超高,仅返回图片前1万像素高度';
$lines[$lines_len-1] = '';
$lines[$lines_len] = '';
$lines = array_slice($lines, 0, $lines_len);
}
$new_image = imagecreatetruecolor($max_width, $image_height);
imagefill($new_image, 0, 0, $white);
imagesavealpha($new_image, true);
$transparent = imagecolorallocatealpha($new_image, 255, 255, 255, 127);
//$backgroundColor = imagecolorallocate($new_image, 255, 255, 255); // 纯白色背景作为示例
imagefill($new_image, 0, 0, $transparent);

// 随机线条参数
$lineCount = 100; // 线条数量
$minLineWidth = 80; // 线条最小宽度
$maxLineWidth = 200; // 线条最大宽度
$minLineColor = 230; // 线条颜色范围下限
$maxLineColor = 255; // 线条颜色范围上限
// 绘制随机线条
for ($i = 0; $i < $lineCount; $i++) {
// 随机选择线条的起点和终点
$startX = rand(-50, $max_width+50);
$startY = rand(-50, $image_height+50);
$endX = rand(-50, $max_width+50);
$endY = rand(-50, $image_height+50);

// 随机选择线条的颜色和宽度
$lineColor = imagecolorallocate($new_image, rand($minLineColor, $maxLineColor), rand($minLineColor, $maxLineColor), rand($minLineColor, $maxLineColor));
$lineWidth = rand($minLineWidth, $maxLineWidth);
// 设置线条宽度
imagesetthickness($new_image, $lineWidth);
// 绘制线条
imageline($new_image, $startX, $startY, $endX, $endY, $lineColor);
}

// 在新图像上绘制文字
foreach ($lines as $i => $line) {
imagettftext($new_image, $font_size, 0, 10, ($i + 1) * $line_height-5, $black, $font_file, $line);
}

// 绘制图片到画布上
if(count($images) > 0){
foreach ($images as $imageInfo) {
// 计算图片在画布上的位置,这里以简单方式为例,实际可能需要更复杂的布局逻辑
$imagePath = $imageInfo['src'];
$imageX = $imageInfo['x'];
$imageY = $imageInfo['y'];
$imageW = $imageInfo['w'];
$imageH = $imageInfo['h'];
$imageNW = $imageInfo['nw'];
$imageNH = $imageInfo['nh'];
// 加载并复制图片到新图像
$imageToDraw = @imagecreatefromstring(file_get_contents($imagePath));
imagecopyresampled($new_image, $imageToDraw, $imageX, $imageY, 0, 0, $imageNW, $imageNH, $imageW, $imageH);
imagedestroy($imageToDraw); // 清理内存
}
}
imagewebp($new_image,$outimg,'80');
imagedestroy($new_image);
$outimg = site_url().'/conimg/'.basename($outimg);
$outimg = str_replace(ABSPATH,site_url(),$outimg);
return $outimg;
}

 

在内容页中添加

<?php if ( is_single() ) { ?>
<img src="<?php echo ii_get_content2image('650','14'); ?>" />
<?php } ?>

<img src="<?php echo ii_get_content2image('650','14'); ?>" />

 

 

 

 

 

Winodws 7/8与Server 2008/2012等系统frp服务无法启动的原因

谷歌Go语言发布1.21更新:放弃了对Winodws 7/8与Server 2008/2012,以及macOS 10.13和10.14的支持。

v0.51.3 开始支持 go 1.21
v0.55.0 支持的最低Go版本已更新为 1.22

新版frp不支持win7,建议win10及以上系统。

win7可以考虑下载v0.51.2版本及之前的版本。

报错信息示例:

Exception 0xc0000005 0x8 0x0 0x0
PC=0x0

runtime.asmstdcall(0x22fcf8)
runtime/sys_windows_amd64.s:75 +0x7a fp=0x22fc80 sp=0x22fc60 pc=0x47a37a

rax 0x0
rbx 0x11e2e78
rcx 0x126afe8
rdx 0x20
rdi 0x7fffffde000
rsi 0x22fe78
rbp 0x22fdc0
rsp 0x22fc58
r8 0x0
r9 0x22fee8
r10 0x13da2e8
r11 0xc000004000
r12 0x22fec8
r13 0x0
r14 0x11e1ec0
r15 0x3
rip 0x0
rflags 0x10293
cs 0x33
fs 0x53
gs 0x2b

解决steam注册问题

steam注册一直要求验证是真人。无限验证,不能通过。提示:

“您对 CAPTCHA 的响应似乎无效,请在下方重新验证您不是机器人”

有同学想了一下,用VPN,然后又提示:

请停用VPN再重新注册。

最后通过测试,得出一个方法:

0.使用网页注册

1.切换语言为英语

2.切换地区为Taiwan

然后顺利通过注册,进邮箱激活一下即可。

 

 

 

 

frp服务端配置文件frps.toml参考

提示:新版frp不支持win7,建议win10及以上系统。win7可以考虑下载0.49.0版尝试。

frp服务端配置文件frps.toml参考

bindAddr = "10.1.5.1"
bindPort = 7000
kcpBindPort = 7000
vhostHTTPPort = 8080
vhostHTTPSPort = 8081
subDomainHost = "xxxx.net"
webServer.addr = "0.0.0.0"
webServer.port = 7500
webServer.user = "user"
webServer.password = "xxxx"
transport.maxPoolCount = 50
auth.token = "xxxx"
custom404Page = "/frps/404.html"

frps.service文件及路径

/etc/systemd/system/frps.service

[Unit]
Description = Frp Server Service
After = network.target syslog.target
Wants = network.target

[Service]
Type = simple
User=root
Group=root
Restart = on-failure
RestartSec = 5s
ExecStart = /frps/frps -c /frps/frps.toml
LimitNOFILE = 1048576

[Install]
WantedBy = multi-user.target

 

frp客户端配置文件frpc.toml参考

提示:新版frp不支持win7,建议win10及以上系统。win7可以考虑下载0.49.0版尝试。

frp客户端配置文件frpc.toml参考

serverAddr = "x.x.x.x"
serverPort = 7000
auth.token = "xxxx"
webServer.addr = "127.0.0.1"
webServer.port = 7400
webServer.user = "xxxx"
webServer.password = "xxxx"
transport.tls.enable = true
transport.tls.disableCustomTLSFirstByte = true

[[proxies]]
name = "RDP"
type = "tcp"
localIP = "127.0.0.1"
localPort = 3389
remotePort = xx89
transport.useEncryption = true
transport.useCompression = true

[[proxies]]
name = "webdav"
type = "tcp"
localIP = "127.0.0.1"
localPort = 9999
remotePort = xx99
transport.useEncryption = true
transport.useCompression = true

[[proxies]]
name = "frpc"
type = "http"
localIP = "127.0.0.1"
localPort = 7400
subdomain = "frpc"
transport.useEncryption = true
transport.useCompression = true

{{- range $_, $v := parseNumberRangePair "20,21,xx01-x03" "xx1,xx1,xx01-xx03" }}
[[proxies]]
name = "ftp-{{ $v.First }}"
type = "tcp"
localIP = "127.0.0.1"
localPort = {{ $v.First }}
remotePort = {{ $v.Second }}
transport.useEncryption = true
transport.useCompression = true
{{- end }}

本例子中含:常用的远程服务RDP,文件系统webdav以及ftp和网页服务http.

 

wordpress添加post_type自定义内容类型

wordpress默认的内容模型只有文章,如果想增加其它的,就要使用到  post_type,通过post_type自定义内容模型,例如影视,图片,链接等模型。

只需在主题的function.php文件中添加对应post_type代码即可。

相关注释说明:

# 在 'init' 钩子上注册自定义文章类型.
add_action('init', 'my_register_post_types');

/**
* 注册插件需要的文章类型
*
* @since 1.0.0
* @access public
* @return void
*/
function my_register_post_types()
{

// 设置文章类型参数
$args = (

// 文章类型的简介,貌似没有在 WordPress 内核中使用,不过我们可以在主题或插件中使用
'description' => __('This is a description for my post type.', 'wprs'),
// 字符串

// 文章类型是否公开给管理员或者前端用户使用,这个参数的值是后面很多参数的默认值
'public' => true,
// bool (default is FALSE)

// 是否可以在前端作为 parse_request() 的一部分查询该文章类型
'publicly_queryable' => true,
// bool (默认为 'public' 参数的值).

// 是否在前端搜索中隐藏该文章类型
'exclude_from_search' => false,
// bool (默认为 'public' 反值)

// 是否可以在导航菜单中选择
'show_in_nav_menus' => false,
// bool (默认为 'public' 参数的值)

// 是否在管理界面生成默认的管理界面,使用后面的参数,可以控制生成的 UI 组件,如果我们要构建自己的管理界面,
//设置该参数为 False
'show_ui' => true,
// bool (默认为 'public' 的值)

// 是否在管理菜单中显示,'show_ui' 参数必须设置为 True,这个参数才有效,我们页可以设置该参数为一个顶级菜单
//(如:'tools.php'),这种情况下,该文章类型的管理菜单出现在 Tools 菜单下面
'show_in_menu' => true,
// bool (默认为 'show_ui' 的值)

// 是否在管理工具条中显示该文章类型,如果设置为 true,WordPress 会在管理工具条中添加一个新建该文章类型文章的链接
'show_in_admin_bar' => true,
// bool (默认为 'show_in_menu' 的值)

// 该文章类型在管理菜单中出现的位置,'show_in_menu' 必须设置为 true,该参数才有用
'menu_position' => null,
// int (默认为 25 - 出现在「评论」菜单后面)

// 管理菜单的图标 URI,或者 Dashicon 的类名称. 参见: https://developer.wordpress.org/resource/dashicons/
'menu_icon' => null,
// 字符串 (默认使用文章图标)

// 属于该文章类型的文章是否可以通过 WordPress 导入/导出插件或者类型的插件导出
'can_export' => true,
// bool (默认为 TRUE)

// 是否暴露在 Rest API 中
'show_in_rest',
// 布尔值,默认为 false

// 使用 Rest API 访问的基础 URI 别名
'rest_base',
// 字符串,默认为文章类型别名

// 使用自定义 Rest API 控制器而不是默认的 WP_REST_Posts_Controller,自定义控制器必须继承 WP_REST_Controller
'rest_controller_class',
// 字符串,默认为 WP_REST_Posts_Controller

// 是否在删除用户时,删除他们撰写的文章
'delete_with_user' => false,
// bool (如果文章类型支持 ‘author’ 功能,该参数默认为 TRUE)

// 该文章类型是否支持多级文章(父级文章/子文章/等等.)
'hierarchical' => false,
// bool (默认为 FALSE)

// 是否为该文章类型开启存档页面 index/archive/root 页面,如果设置为 TRUE, 该文章类型名称将作为存档页面别名使用,
//当然,我们页可以设置自定义存档别名
'has_archive' => 'example',
// bool|string (默认为 FALSE)

// 为该文章类型设置 query_var 键,如果设置为 TRUE, 将使用文章类型名称,如果需要,也可以设置自定义字符串
'query_var' => 'example',
// bool|string (默认为 TRUE - 文章类型名称)

// 用于构建该文章类型的编辑、删除、阅读权限的字符串,可以设置字符串或者数组,如果单词的负数不是加“s”的形式,我们需要
//设置一个数组,array( 'box', 'boxes' )
'capability_type' => 'example',
// string|array (默认为 'post')

// 是否让 WordPress 映射权限元数据 (edit_post, read_post, delete_post),如果设置为 FALSE, 我们需要自己通过
//过滤 “map_meta_cap” 钩子来设置文章类型权限
'map_meta_cap' => true,
// bool (默认为 FALSE)

// 设置更精确的文章类型权限,WordPress 默认使用 'capability_type' 参数来构建权限,多数情况下,我们不需要像文章
//或页面这么完整的权限,下面是我经常使用的几个权限: 'manage_examples', 'edit_examples', 'create_examples'.
// 每个文章类型都是独特的,我们可以根据需要调整这些权限
'capabilities' => (

// meta caps (don't assign these to roles)
'edit_post' => 'edit_example',
'read_post' => 'read_example',
'delete_post' => 'delete_example',

// primitive/meta caps
'create_posts' => 'create_examples',

// primitive caps used outside of map_meta_cap()
'edit_posts' => 'edit_examples',
'edit_others_posts' => 'manage_examples',
'publish_posts' => 'manage_examples',
'read_private_posts' => 'read',

// primitive caps used inside of map_meta_cap()
'read' => 'read',
'delete_posts' => 'manage_examples',
'delete_private_posts' => 'manage_examples',
'delete_published_posts' => 'manage_examples',
'delete_others_posts' => 'manage_examples',
'edit_private_posts' => 'edit_examples',
'edit_published_posts' => 'edit_examples',
),

// 定义该文章类型的 URL 结构,我们可以设置一个具体的参数或一个布尔值,如果设置为 false,该文章类型将不支持
// URL Rewrite 功能
'rewrite' => (

// 文章类型的别名
'slug' => 'example', // string (默认为文章类型名称)

// 是否在固定链接中显示 $wp_rewrite->front 文章类型别名
'with_front' => false, // bool (默认为 TRUE)

// 是否允许文章类型中的文章通过 <!--nextpage--> 快捷标签实现分页
'pages' => true, // bool (默认为 TRUE)

// 是否为订阅源创建漂亮的固定链接feeds.
'feeds' => true, // bool (默认为 'has_archive' 的值)

// 为固定链接设置设置 endpoint 遮罩
'ep_mask' => EP_PERMALINK, // const (默认为 EP_PERMALINK)
),

// 文章类型支持的 WordPress 功能,许多参数在文章编辑界面非常有用。这有助于其他主题和插件决定让用户使用什么功能
//或者提供什么数据,我们可以为该参数设置一个数组,也可以设置为 false,以防止添加任何功能,文章类型创建后,我们
//可以使用 add_post_type_support() 添加功能,或使用 remove_post_type_support() 删除功能。默认功能是“标题
//”和“编辑器”。
'supports' => (
'title',// 文章标题 ($post->post_title).
'editor', // 文章内容 ($post->post_content).
'excerpt', // 文章摘要 ($post->post_excerpt).
'author', // 文章作者 ($post->post_author).
'thumbnail',// 特色图像 (当前站点使用的主题必须支持 'post-thumbnails').
'comments', // 显示评论元数据盒子,如果设置了该值, 这个文章类型将支持评论
'trackbacks', // 在编辑界面显示允许发送链接通知的元数据盒子
'custom-fields', // 显示自定义字段元数据盒子
'revisions', // 显示版本元数据盒子,如果设置了该参数,WordPress 将在数据库中保存文章版本
'page-attributes', // 显示“页面属性”元数据盒子,包含父级页面或页面排序字段。
'post-formats',// 显示文章格式元数据盒子,并允许该文章类型使用文章格式
),
// 标签用来在管理界面或前端显示该文章类型的名称,标签参数不会自动改写文章更新、错误等信息中的字段,我们需要过滤
// 'post_updated_messages' 钩子来自定义这些消息。
'labels' => (
'name' => __('Posts', 'wprs'),
'singular_name' => __('Post', 'wprs'),
'menu_name' => __('Posts', 'wprs'),
'name_admin_bar' => __('Posts', 'wprs'),
'add_new' => __('Add New', 'wprs'),
'add_new_item' => __('Add New Post', 'wprs'),
'edit_item' => __('Edit Post', 'wprs'),
'new_item' => __('New Post', 'wprs'),
'view_item' => __('View Post', 'wprs'),
'search_items' => __('Search Posts', 'wprs'),
'not_found' => __('No posts found', 'wprs'),
'not_found_in_trash' => __('No posts found in trash', 'wprs'),
'all_items' => __('All Posts', 'wprs'),
'featured_image' => __('Featured Image', 'wprs'),
'set_featured_image' => __('Set featured image', 'wprs'),
'remove_featured_image' => __('Remove featured image', 'wprs'),
'use_featured_image' => __('Use as featred image', 'wprs'),
'insert_into_item' => __('Insert into post', 'wprs'),
'uploaded_to_this_item' => __('Uploaded to this post', 'wprs'),
'views' => __('Filter posts list', 'wprs'),
'pagination' => __('Posts list navigation', 'wprs'),
'list' => __('Posts list', 'wprs'),

// 只在分级文章类型中使用的标签
'parent_item' => __('Parent Post', 'wprs'),
'parent_item_colon' => __('Parent Post:', 'wprs'),
),
);

// 注册文章类型
register_post_type(
'example', // 文章类型名称,最多 20 个字符,不支持大写或空格
$args // 文章类型的参数
);

}

post_type自定义产品模型参考:

// Register Custom Post Type
function products_post_type() {

$labels = array(
'name' => _x( 'Products', 'Post Type General Name', 'text_domain' ),
'singular_name' => _x( 'Product', 'Post Type Singular Name', 'text_domain' ),
'menu_name' => __( 'Products', 'text_domain' ),
'name_admin_bar' => __( 'Product', 'text_domain' ),
'archives' => __( 'Item Archives', 'text_domain' ),
'attributes' => __( 'Item Attributes', 'text_domain' ),
'parent_item_colon' => __( 'Parent Product:', 'text_domain' ),
'all_items' => __( 'All Products', 'text_domain' ),
'add_new_item' => __( 'Add New Product', 'text_domain' ),
'add_new' => __( 'New Product', 'text_domain' ),
'new_item' => __( 'New Item', 'text_domain' ),
'edit_item' => __( 'Edit Product', 'text_domain' ),
'update_item' => __( 'Update Product', 'text_domain' ),
'view_item' => __( 'View Product', 'text_domain' ),
'view_items' => __( 'View Items', 'text_domain' ),
'search_items' => __( 'Search products', 'text_domain' ),
'not_found' => __( 'No products found', 'text_domain' ),
'not_found_in_trash' => __( 'No products found in Trash', 'text_domain' ),
'featured_image' => __( 'Featured Image', 'text_domain' ),
'set_featured_image' => __( 'Set featured image', 'text_domain' ),
'remove_featured_image' => __( 'Remove featured image', 'text_domain' ),
'use_featured_image' => __( 'Use as featured image', 'text_domain' ),
'insert_into_item' => __( 'Insert into item', 'text_domain' ),
'uploaded_to_this_item' => __( 'Uploaded to this item', 'text_domain' ),
'items_list' => __( 'Items list', 'text_domain' ),
'items_list_navigation' => __( 'Items list navigation', 'text_domain' ),
'filter_items_list' => __( 'Filter items list', 'text_domain' ),
);
$args = array(
'label' => __( 'Product', 'text_domain' ),
'description' => __( 'Product information pages.', 'text_domain' ),
'labels' => $labels,
'supports' => array( 'title', 'editor', 'thumbnail', 'comments', 'custom-fields' ),
'taxonomies' => array( 'category', 'post_tag' ),
'hierarchical' => false,
'public' => true,
'show_ui' => true,
'show_in_menu' => true,
'menu_position' => 5,
'show_in_admin_bar' => true,
'show_in_nav_menus' => true,
'can_export' => true,
'has_archive' => true,
'exclude_from_search' => false,
'publicly_queryable' => true,
'capability_type' => 'page',
);
register_post_type( 'product', $args );

}
add_action( 'init', 'products_post_type', 0 );

文章模型参考:

// Register Custom Post Type
function articles_post_type() {

$labels = array(
'name' => _x( 'Articles', 'Post Type General Name', 'text_domain' ),
'singular_name' => _x( 'Article', 'Post Type Singular Name', 'text_domain' ),
'menu_name' => __( 'Articles', 'text_domain' ),
'name_admin_bar' => __( 'Article', 'text_domain' ),
'archives' => __( 'Item Archives', 'text_domain' ),
'attributes' => __( 'Item Attributes', 'text_domain' ),
'parent_item_colon' => __( 'Parent Item:', 'text_domain' ),
'all_items' => __( 'All Items', 'text_domain' ),
'add_new_item' => __( 'Add New Item', 'text_domain' ),
'add_new' => __( 'Add New', 'text_domain' ),
'new_item' => __( 'New Item', 'text_domain' ),
'edit_item' => __( 'Edit Item', 'text_domain' ),
'update_item' => __( 'Update Item', 'text_domain' ),
'view_item' => __( 'View Item', 'text_domain' ),
'view_items' => __( 'View Items', 'text_domain' ),
'search_items' => __( 'Search Item', 'text_domain' ),
'not_found' => __( 'Not found', 'text_domain' ),
'not_found_in_trash' => __( 'Not found in Trash', 'text_domain' ),
'featured_image' => __( 'Featured Image', 'text_domain' ),
'set_featured_image' => __( 'Set featured image', 'text_domain' ),
'remove_featured_image' => __( 'Remove featured image', 'text_domain' ),
'use_featured_image' => __( 'Use as featured image', 'text_domain' ),
'insert_into_item' => __( 'Insert into item', 'text_domain' ),
'uploaded_to_this_item' => __( 'Uploaded to this item', 'text_domain' ),
'items_list' => __( 'Items list', 'text_domain' ),
'items_list_navigation' => __( 'Items list navigation', 'text_domain' ),
'filter_items_list' => __( 'Filter items list', 'text_domain' ),
);
$args = array(
'label' => __( 'Article', 'text_domain' ),
'description' => __( 'Site articles.', 'text_domain' ),
'labels' => $labels,
'supports' => array( 'title', 'editor', 'thumbnail', 'comments', 'revisions' ),
'taxonomies' => array( 'category', 'post_tag' ),
'hierarchical' => false,
'public' => true,
'show_ui' => true,
'show_in_menu' => true,
'menu_position' => 5,
'show_in_admin_bar' => true,
'show_in_nav_menus' => true,
'can_export' => true,
'has_archive' => true,
'exclude_from_search' => false,
'publicly_queryable' => true,
'capability_type' => 'page',
);
register_post_type( 'articles', $args );

}
add_action( 'init', 'articles_post_type', 0 );

添加自定义 Meta Box 需要用到 add_meta_box 函数,注册一个 Meta Box :

add_action( 'add_meta_boxes', 'movie_director' );
function movie_director() {
add_meta_box(
'movie_director',
'电影导演',
'movie_director_meta_box',
'movie',
'side',
'low'
);
}

然后在配置参数里面指定了回调函数 movie_director_meta_box,我们需要在这个函数里面创建表单:

function movie_director_meta_box($post) {
// 创建临时隐藏表单,为了安全
wp_nonce_field( 'movie_director_meta_box', 'movie_director_meta_box_nonce' );
// 获取之前存储的值
$value = get_post_meta( $post->ID, '_movie_director', true );
?>
<label for="movie_director"></label>
<input type="text" id="movie_director" name="movie_director" value="" placeholder="输入导演名称" >
<?php
}

自定义 Meta Box 数据的保存

add_action( 'save_post', 'movie_director_save_meta_box' );
function movie_director_save_meta_box($post_id){
// 安全检查
// 检查是否发送了一次性隐藏表单内容(判断是否为第三者模拟提交)
if ( ! isset( $_POST['movie_director_meta_box_nonce'] ) ) {
return;
}
// 判断隐藏表单的值与之前是否相同
if ( ! wp_verify_nonce( $_POST['movie_director_meta_box_nonce'], 'movie_director_meta_box' ) ) {
return;
}
// 判断该用户是否有权限
if ( ! current_user_can( 'edit_post', $post_id ) ) {
return;
}
// 判断 Meta Box 是否为空
if ( ! isset( $_POST['movie_director'] ) ) {
return;
}
$movie_director = sanitize_text_field( $_POST['movie_director'] );
update_post_meta( $post_id, '_movie_director', $movie_director );
}

输出自定义 Meta Box 数据参考:

echo '导演:'.get_post_meta( get_the_ID(), '_movie_director', true );

后台列表中显示更多字段,使用 manage_$post_type_posts_custom_column 即可实现

add_action("manage_posts_custom_column", "movie_custom_columns");
add_filter("manage_edit-movie_columns", "movie_edit_columns");
function movie_custom_columns($column){
global $post;
switch ($column) {
case "movie_director":
echo get_post_meta( $post->ID, '_movie_director', true );
break;
}
}
function movie_edit_columns($columns){
$columns['movie_director'] = '导演';
return $columns;
}

即添加了列导演字段,并从每篇文章中读取出来。

 

调用自定义post_type内容模型方法,需要在主题function.php文件添加 pre_get_posts 这个 action 处理:

add_action( 'pre_get_posts', 'add_my_post_types_to_query' );
function add_my_post_types_to_query( $query ) {
if ( is_home() && $query->is_main_query() )
$query->set( 'post_type', array( 'post', 'page', 'movie' ) );
return $query;
}

在上面的 $query 变量里面设置的 post_type 数组就是要在主循环里面展示的内容,将你的自定义 Post Type 填写进去就可以在首页中显示出来了。

新建product的post_type模板archive-product.php,代码参考:

<?php $args = array( 'post_type' => 'product', 'posts_per_page' => 10);
$loop = new WP_Query( $args );
while ( $loop->have_posts() ) : $loop->the_post(); ?>
<div class="col-4">
<a href="<?php the_permalink(); ?>" class="item wow zoomIn"> 
<b><?php the_title(); ?></b>
</div>
</a>
</div>
<?php endwhile; ?>
<?php
the_posts_pagination( array(
'mid_size' => 2,
'prev_text' => __( 'Prev', 'textdomain' ),
'next_text' => __( 'Next', 'textdomain' ),
) );
?>

本文参考:

https://www.cnblogs.com/ytkah/p/11868158.html

https://www.cnblogs.com/ytkah/p/11926186.html

https://blog.wpjam.com/article/wordpress-post-type/

 

pdf.js报错MIME type of "application/octet-stream"的解决方法

版本:pdfjs-4.0.379-legacy-dist

原因,新版pdf.js有properties,bcmap和mjs三种格式文件,默认情况下,服务器环境不支持,需要单独配置对应的MIME类型。

解决方法:

服务器配置MIME类型。

nginx服务器如果配置文件中有

http
{
include mime.types;

就可以直接在mime.types中添加三行:

application/javascript mjs;
application/octet-stream properties;
application/octet-stream bcmap;

其它类型服务器可以参考添加。

FRP专用反代配置备份

提示:新版frp不支持win7,建议win10及以上系统。win7可以考虑下载0.49.0版尝试。

http
{
......
resolver 1.1.1.1;#添加这一行,避免出现502错误
......

## FRP专用反代配置开始

server
{
listen 80;
server_name *.xxxx.net;
location / {
proxy_pass http://$host:8080;
proxy_set_header Host $host:80;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_hide_header X-Powered-By;
}
location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$
{
proxy_pass http://$host:8080;#添加这一行
expires 30d;
}

location ~ .*\.(js|css)?$
{
proxy_pass http://$host:8080;#添加这一行
expires 12h;
}
location ~ /\.
{
deny all;
}
}

server
{
listen 80;
server_name www.xxxx.net;
location / {
proxy_pass http://$host:8080;
proxy_set_header Host $host:80;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_hide_header X-Powered-By;
}
location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$
{
proxy_pass http://$host:8080;#添加这一行
expires 30d;
}

location ~ .*\.(js|css)?$
{
proxy_pass http://$host:8080;#添加这一行
expires 12h;
}
location ~ /\.
{
deny all;
}
}

server
{
listen 80;
listen 443 ssl http2;
server_name yun.xxxxx.net;
large_client_header_buffers 4 1m;

location / {
proxy_pass http://$host:8080;
proxy_set_header Host $host:80;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_hide_header X-Powered-By;
proxy_buffer_size 64k;
proxy_buffers 4 32k;
proxy_busy_buffers_size 64k;
}

ssl_certificate /frps/yun.xxxx.net/fullchain.pem;
ssl_certificate_key /frps/yun.xxxx.net/privkey.pem;
ssl_protocols TLSv1.1 TLSv1.2 TLSv1.3;
ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
ssl_prefer_server_ciphers on;
ssl_session_cache shared:SSL:10m;
ssl_session_timeout 10m;
error_page 497 https://$host$request_uri;

#SSL-END

location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$
{
proxy_pass http://$host:8080;#添加这一行
expires 30d;
}

location ~ .*\.(js|css)?$
{
proxy_pass http://$host:8080;#添加这一行
expires 12h;
}
location ~ /\.
{
deny all;
}
}

## FRP专用反代配置结束