30 十二月, 2023

thumbnail

盒子加外置硬盘搭建Navidrome私人音乐服务器

本文资源都来源网络只是整合修改
家里使用PC搭建了NAS配置N3160低功耗U 8快8TBSAS硬盘 平时使用频率并不是很高出于对数据的保护和节能的目的将私有云盘及私人音乐服务器迁移至ARM ubuntu上
arm ubuntu 使用M301H 刷写最新64位系统 外置硬盘使用3.5寸硬盘盒加2TB希捷硬盘 电源使用12V3A 1拖2 5.5MM适配器 经过测试盒子加硬盘盒满载14W 待机2W左右
私有云请参照论坛里可道云的安装此处不在重复下面开始Navidrome安装
1.新建一个组和用户用来运行Navidrome,官方不推荐使用root用户运行Navidrome
groupadd -r navidrome 创建组
useradd -r -g navidrome -s /sbin/nologin -M -c "Navidrome Daemons" navidrome 创建用户

  1. 创建Navidrome的安装目录与运行目录。
    install -d -o navidrome -g navidrome /opt/navidrome 安装目录
    install -d -o navidrome -g navidrome /var/lib/navidrome 运行目录

3.从github下载Navidrome最新的二进制文件,重命名为Navidrome.tar.gz解压至安装目录并且设置运行权限。
https://github.com/navidrome/navidrome/releases github地址
下面是下载地址清按需下载 我刷了64位系统使用arm64 其他的清按需下载
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Linux_arm64.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Linux_armv7.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Linux_armv6.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Linux_armv5.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Linux_i386.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Linux_x86_64.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_macOS_x86_64.tar.gz
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Windows_i386.zip
https://github.com/navidrome/navidrome/releases/download/v0.47.5/navidrome_0.47.5_Windows_x86_64.zip

sudo tar -xvzf Navidrome.tar.gz -C /opt/navidrome/ 解压文件
sudo chown -R navidrome:navidrome /opt/navidrome 给与权限

  1. 创建Navidrome的配置文件
    https://www.navidrome.org/docs/usage/configuration-options/ 官方给出的配置文件可用选项

    vim /var/lib/navidrome/navidrome.toml 创建编辑
    MusicFolder = "<library_path>" 文件内容<library_path>修改为音乐文件存放路径


如果家里有IPV6可添加大于10000以上端口并在路由器上放开对应IPV6端口可实现直接远程连接
port = "32400" 此项不是必须因我使用PLEX路由器以开放32400端口 如无此项默认使用4533端口

  1. 创建启动服务项目
    vim /etc/systemd/system/navidrome.service 创建编辑

下为内容

[Unit]  Description=Navidrome Music Server and Streamer compatible with Subsonic/Airsonic  After=remote-fs.target network.target  AssertPathExists=/var/lib/navidrome    [Install]  WantedBy=multi-user.target    [Service]  User=navidrome  Group=navidrome  Type=simple  ExecStart=/opt/navidrome/navidrome --configfile "/var/lib/navidrome/navidrome.toml"  WorkingDirectory=/var/lib/navidrome  TimeoutStopSec=20  KillMode=process  Restart=on-failure    #See https://www.freedesktop.org/software/systemd/man/systemd.exec.html  DevicePolicy=closed  NoNewPrivileges=yes  PrivateTmp=yes  PrivateUsers=yes  ProtectControlGroups=yes  ProtectKernelModules=yes  ProtectKernelTunables=yes  RestrictAddressFamilies=AF_UNIX AF_INET AF_INET6  RestrictNamespaces=yes  RestrictRealtime=yes  SystemCallFilter=~@clock @debug @module @mount @obsolete @"reboot"#42 @setuid @swap  ReadWritePaths=/var/lib/navidrome    # You can uncomment the following line if you're not using the jukebox This  # will prevent navidrome from accessing any real (physical) devices  #PrivateDevices=yes    # You can change the following line to `strict` instead of `full` if you don't  # want navidrome to be able to write anything on your filesystem outside of  # /var/lib/navidrome.  ProtectSystem=full    # You can comment the following line if you don't have any media in /home/*.  # This will prevent navidrome from ever reading/writing anything there.  #ProtectHome=true

  1. 更新并启动Navidrome
    systemctl daemon-reload 更新服务列表
    systemctl start navidrome.service 启动服务
    systemctl status navidrome.service 查看服务状态

如出现 Active: active (running) 服务以启动
使用 ip:4533 访问服务器
一切正常以后添加开启启动服务
sudo systemctl enable navidrome.service


支持应用
iOS:play:Sub、 substreamer、 Amperfy和 iSub
安卓:DSub, Subtracks, substreamer, Ultrasonic和 Audinaut
网络:Subplayer、 Airsonic Refix、 Aurial、 Jamstash和 Subfire
桌面:Sublime Music (Linux) 和Sonixd (Windows/Linux/macOS)
CLI:Jellycli (Windows/Linux) 和STMP (Linux/macOS)
连接的扬声器:
Sonos: bonob Alexa:AskSonic
其他:
Subsonic Kodi 插件、 Navidrome Kodi 插件、 HTTP目录文件系统

如果是MP3 之类不需要解码可直接使用 如果遇到其它格式播放器无法识别需要转码的需要ffmpeg 下为安装步骤

  1. 安装依赖库
    apt-get update
    apt-get -y install \
    autoconf \
    automake \
    build-essential \
    cmake \
    git-core \
    libass-dev \
    libfreetype6-dev \
    libgnutls28-dev \
    libmp3lame-dev \
    libsdl2-dev \
    libtool \
    libva-dev \
    libvdpau-dev \
    libvorbis-dev \
    libxcb1-dev \
    libxcb-shm0-dev \
    libxcb-xfixes0-dev \
    meson \
    ninja-build \
    pkg-config \
    texinfo \
    wget \
    yasm \
    zlib1g-dev \
    libunistring-dev

  2. 安装ffmpeg
    apt install ffmpeg


    验证安装
    ffmpeg –version


    出现下面提示表示安装完成
    ffmpeg version 4.2.7-0ubuntu0.1 Copyright (c) 2000-2022 the FFmpeg developers
    built with gcc 9 (Ubuntu 9.4.0-1ubuntu120.04.1)
    configuration: --prefix=/usr --extra-version=0ubuntu0.1 --toolchain=hardened --libdir=/usr/lib/aarch64-linux-gnu --incdir=/usr/include/aarch64-linux-gnu --arch=arm64 --enable-gpl --disable-stripping --enable-avresample --disable-filter=resample --enable-avisynth --enable-gnutls --enable-ladspa --enable-libaom --enable-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libcdio --enable-libcodec2 --enable-libflite --enable-libfontconfig --enable-libfreetype --enable-libfribidi --enable-libgme --enable-libgsm --enable-libjack --enable-libmp3lame --enable-libmysofa --enable-libopenjpeg --enable-libopenmpt --enable-libopus --enable-libpulse --enable-librsvg --enable-librubberband --enable-libshine --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libssh --enable-libtheora --enable-libtwolame --enable-libvidstab --enable-libvorbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-libx265 --enable-libxml2 --enable-libxvid --enable-libzmq --enable-libzvbi --enable-lv2 --enable-omx --enable-openal --enable-opencl --enable-opengl --enable-sdl2 --enable-libdc1394 --enable-libdrm --enable-libiec61883 --enable-chromaprint --enable-frei0r --enable-libx264 --enable-shared
    libavutil 56. 31.100 / 56. 31.100
    libavcodec 58. 54.100 / 58. 54.100
    libavformat 58. 29.100 / 58. 29.100
    libavdevice 58. 8.100 / 58. 8.100
    libavfilter 7. 57.100 / 7. 57.100
    libavresample 4. 0. 0 / 4. 0. 0
    libswscale 5. 5.100 / 5. 5.100
    libswresample 3. 5.100 / 3. 5.100
    libpostproc 55. 5.100 / 55. 5.100

 
thumbnail

轻度用户一些刚需功能设置的分享贴

本贴主要包括三个内容:挂载本地硬盘,alist云盘映射到本地,zerotier内网穿透,我也是个小白,现学现卖的弄了一个礼拜,作为轻度用户,刚需功能都搭好了,分享一下经验,很多用语可能很业余,各方面有什么不对的请各路大神多指正。
一、硬盘挂载(使用说明里说的很清楚了,我就简单归纳一下)
1.1先检查磁盘识别情况
指令:df -h
可以看到已识别的硬盘路径,应该是:/mnt/sda1
建立一个新文件夹,名称随意(以usb-share为例)
1.2赋权
指令:chmod -R 777 /mnt/sda1/usb-share
1.3硬盘配置文件使硬盘可见
开启vi编辑修改文件,指令:vi /etc/samba/smb.conf
在文件结尾添加代码(可百度Vi编辑器常用操作)
中括号里是显示在文件管理器中的名字,与文件夹的名字可以不一致。如下所示:虽然我新建的文件夹叫usb-share,但是挂在到本地磁盘后看到的是HH
[HH]
path = /mnt/sda1/usb-share
read only = no
guest ok = yes
create mask = 0777
directory mask = 0777
browseable = yes

1.4重启Samba 服务
指令:systemctl restart smbd

二、挂载云盘
2.1配置alist储存
查看nas主页使用说明教程相应链接。
添加相应云盘:个人常用:阿里云盘、天翼云盘

2.2安装davfs2
此步骤为我误入歧途时查的一个教程中让安装的,为在nas端挂载并分享alist用,之后发现该方法不可行,但是换方法后也没有卸载,所以我也不知道是不是必须安装。我个人觉得他不是必须的,可以省略,但以防万一还是写出来吧。
指令:sudo apt-get install davfs2
2.3云盘映射到本地
选择相应的软件设置,推荐:
手机端 猫头鹰管理器(我用小米手机,应用商店里就能下载,我觉得还挺好用的,华为如果是本地访问可以用自带的文件管理器,但是穿透的时候好像不行)
电脑端 airlivedrive

云盘路径:默认alist开放端口为5244,路径为/alist/dav
地址格式为:http://192.168..:5244/alist/dav/
账号密码:alist中自行设置
相应软件按上述信息填写即可
PS:添加天翼云盘时要选本地代理而非302定向,映射磁盘的时候才有读写权限。但是天翼云盘的权限问题受映射软件影响,如在PC端的使用raidrive映射后,阿里云盘正常使用,但是天翼云盘只能看见文件名却无读写权限,改本地代理后也只有读取无法写入。所以我改成了airlivedrive这个软件。
直接百度软件名字,就有官网下载就行了。

三、免费内网穿透(zerotier)
3.1zerotier组网
建议自行百度图文教程,真的很简单,注册个账号的事儿,建议先用手机和电脑试一下,因为直接用Linux的加入组网出了问题你可能也不知道是怎么回事儿,反正最终也是要多终端打通,所以用手机电脑先组上,对组网的整个流程有个概念,后边省事一点儿。
3.2 Linux系统下安装本软件
运行官网里面给的linux系统安装指令
curl -s https://install.zerotier.com | sudo bash
这个环节一般没问题,出了问题就重置或者百度吧。
3.3加入 提前建立好的zerotier组网id(###########)
指令:sudo zerotier-cli join ###########
正常则返回:200 join OK
之后去zerotier官网登录自己的账号进入该id的管理界面找到nas这个设备打勾就行了。看不懂本段的建议你如4.1所说先去用电脑和手机组个网。
打完勾输入分配给nas的ip就可以远程访问了。有一个开机自动启动的指令,但是它这个应该是默认就已经打开开机启动了的,反正我nas断电了好几次都能直接连上。
3.4 Linux端zerotier常用指令
zerotier-cli status:查看连接状态,正常则返回200 info ########### 1.10.6 ONLINE、异常则返回200 info ########### 1.10.6 OFFLINE
sudo zerotier-cli listnetworks:查看当前连接的网络,返回两行内容,代表当前连接网络的运行状态,要关注的是status这个参数,当其显示:ok则说明已连接;acdess 表示在管理页面可以看见设备但是没打勾;REQUESTING_CONFIGURATION 表示它网你的zerotier服务器发送组网申请了,但是没发出去,服务器也没收到这个申请,所以不显示新设备。


 
thumbnail

nclone实战

.. 这是一个nclone的实战说明

========================================================================

nclone实战

:作者: holts
:Email: [email protected]
:版本: $Revision: 1.00 $
:日期: $Date: 2023/12/12 16:28:00 $
:声明: 不承担由此产生的一切不良后果
:版权: 此文档遵从开放原则,你可以自由的复制,修改,发布,发售,购买, 租借,销毁,改进;或是引用,摘录,合并,比较,分类等等, 总之你可以自由使用此文档.

.. contents:: 内容索引 Table of Contents
.. section-numbering::

rclone安装

========================================================================

Rclone 官网: https://rclone.org/
负责把网盘的webdav映射到NAS的文件系统内

sudo -v ; curl https://rclone.org/install.sh | sudo bash
sudo apt-get install fuse
sudo ln -s /bin/fusermount /bin/fusermount3

配置

========================================================================

sudo rclone config

按提示为remote取名,这里我以alist为名字.
上述操作完成之后会列出一个支持列表,选择存储类型,此处选择webdav,从列出的选项可知输入52(注意:根据版本的不同,此处的数字可能会有变化,选择webdav对应的编号即可,请自行选择自己机器上列出的选项)

接下来填写webdav的地址,输入webdav的地址,此处使用的是alist的webdav地址,请根据自己安装的alist进行填写,这里填写 Http://xxx.ltd/alist/dav/

选择供应商,填写other即可, 接下来输入用户名和密码, 这个就是alist按装后的管理员名和客码.

后续按默认值一路回车完成配置文件和alist的绑定.

生成的配置文件保存在/root/.config/rclone/rclone.conf

通过alist挂载

========================================================================

alist支持国内大部分网盘转为webdav服务,而rclone可以挂载webdav为本地硬盘, 两者组合实现了各种网盘变为本地硬盘, 在alist中wedav策略可以选302重定向或者本地代理, 如果webav访问对象和alist是局域网,那么两种模式速度没什么区别, 建议选代理模式.

如果是外网的话302跳转可以绕过服务器直接访问网盘, 速度要更快也不占用alist服务器带宽, 不过302跳转可能要设置请求header来适配各家网盘.

开始的挂载数据目录路径为:/mnt/dav/, 在 /mnt/dav/ 目录下创建一个文件夹cloud,用于存放rclone挂载的alist,并给权限

sudo mkdir /mnt/dav/cloud/
sudo chown holts:family /mnt/dav/cloud/
sudo chmod 777 /mnt/dav/cloud/

sudo rclone mount alist:/ /mnt/dav/cloud --cache-dir /tmp --allow-other --vfs-cache-mode writes

开机自启

========================================================================

通过systemctl方式来实现,在 /etc/systemd/system/ 目录下创建一个 rclone.service

cat > /etc/systemd/system/rclone.service <<EOF
[Unit]
Description=Rclone
AssertPathIsDirectory=LocalFolder
After=network-online.target
[Service]
Type=simple
ExecStart=rclone mount alist:/ /mnt/dav/cloud --cache-dir /tmp --allow-other --vfs-cache-mode writes
ExecStop=fusermount -qzu LocalFolder
Restart=on-abort
User=root
[Install]
WantedBy=multi-user.target
EOF

#开始启动
systemctl start rclone
#设置开机自启
systemctl enable rclone
#其他使用
systemctl restart rclone #重启
systemctl stop rclone #停止
systemctl status rclone #状态


 
thumbnail

免费蒲公英pgy组网安装教程

蒲公英智能组网VPN客户端目前已支持在树莓派操作系统上使用,可在蒲公英官网下载页面,根据对应的系统版本进行下载。Raspberry Pi 64位 当前版本:2.4.0.52291
deb package:
wget https://pgy.oray.com/softwares/58/download/1839/PgyVisitor_Raspberry_2.4.0.52291_arm64.deb

注意:蒲公英VPN客户端 for 树莓派安装步骤都需要在管理员权限下运行。

1.安装
下载安装包后,通过cd命令进入对应目录找到蒲公英的安装包,输入以下命令进行安装。 dpkg -i PgyVisitor_Raspberry_2.4.0.52291_arm64.deb

  1. 帮助命令
    安装成功后,在系统的任意路径下,输入pgyvisitor -h/pgyvisitor命令并回车,显示所有支持的子命令及其说明。

输入pgyvisitor -v命令并回车,显示当前蒲公英客户端软件版本号。

  1. 子命令
    3.1 登录
    输入pgyvisitor login -h命令并回车,显示登录命令支持的参数及其说明。

3.1.1 交互登录
输入pgyvisitor login/pgyvisitor login -u [UID]命令并回车,进行交互登录。

3.1.2 带参数登录
输入pgyvisitor login -u [UID] -p [password]命令并回车,进行带参数登录。

3.2 退出
输入pgyvisitor logout命令并回车,退出当前的蒲公英UID登录。

3.3 历史登录信息
输入pgyvisitor logininfo命令并回车,显示登录UID的历史登录设备信息。

3.4 自动登录
输入pgyvisitor autologin -h命令并回车,显示自动登录命令支持的参数及其说明。

3.4.1 开启自动登录
输入pgyvisitor autologin -y命令并回车,开启自动登录蒲公英客户端。当设备开机或重启后,蒲公英客户端将自动登录。

3.4.2 关闭自动登录
输入pgyvisitor autologin -n命令并回车,关闭自动登录蒲公英客户端。

3.5 旁路信息
输入pgyvisitor bypass命令并回车,显示组网内其他成员的旁路信息。

3.6 组网成员
输入pgyvisitor getmbrs/pgyvisitor getmbrs -h命令并回车,显示查看组网成员命令支持的参数及其说明。

3.6.1 查看组网信息
输入pgyvisitor getmbrs -m命令并回车,显示当前组网名称、在线成员的信息及离线成员的信息。

3.6.2 查询子设备
输入pgyvisitor getmbrs -s [在线且有子设备的硬件成员序号]命令并回车,查看该硬件成员下的子设备信息。

3.7 查看客户端信息
输入pgyvisitor showsets命令并回车,显示当前登录的蒲公英客户端UID以及自动登录的状态。

  1. 查看日志
    蒲公英客户端日志文件路径:/var/log/oray/pgyvpn

  2. 卸载
    蒲公英客户端卸载命令:dpkg -r pgyvpn


 
thumbnail

利用Docker 搭建 calibre-web看电子书,收藏自己喜欢的书,构建个人图书馆


docker run -d \
--name=calibre-web \
--restart=always \
-p 8083:8083 \
-v /home/docker/calibre-web/config:/config \
-v /home/docker/calibre-web/library:/library \
--privileged \
johngong/calibre-web

默认账号密码: admin/admin123

通过 8083 端口登录后记得进行修改密码

搭建完以后及时给管理员添加权限,特别是上传和阅读的权限,给guest账户添加阅读权限

很好很不错,特别是epub格式的电子书

 

25 十二月, 2023

thumbnail

海纳斯桌面——图标自定义、背景自定义功能扩展

海纳斯桌面——图标自定义、背景自定义功能扩展(只需两步就能一劳永逸)

github地址:https://github.com/825641748/hinas_table/blob/main/README.md
文件下载地址:
add.png
controller.php
index.html

1、使用项目中的相应文件替换原有文件,增加controller.php

index.html替换/var/www/html/index.html
controller.php上传到目录/var/www/html/
add.png上传到目录/var/www/html/img/png/
image
image

2、为文件夹增加写入权限

root@hi3798mv300:~# chmod 777 /var/www/html/icons_wan/  root@hi3798mv300:~# chmod 777 /var/www/html/icons_lan/  root@hi3798mv300:~# chmod 777 /var/www/html/img/png/

image

大功告成!接下来看看详细的使用说明吧

效果图

image

添加示例

image
image
image

参数说明

image

长按自定义图标可以删除,为防止误操作,默认图标不可删

image

已优化原有页面样式,自定义图标理论上无数量限制

长按添加图标上传背景图

e5a359331033e8b6408b6624cc0a5a6
4243c2aa1c53bb076a01474495d234d

如果无法直接下载可以复制文件内容,自行创建文件
index.html

<!DOCTYPE html>  <html>    	<head>  		<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />  		<meta http-equiv="X-UA-Compatible" content="IE=edge">  		<meta name="renderer" content="webkit">  		<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">  		<link rel="shortcut icon" href="/favicon.ico">  		<link rel="stylesheet" type="text/css" href="css/style.css" />  		<link rel="stylesheet" type="text/css" href="css/bg.css" />  		<title>海纳思系统</title>  		<style>  			.adleft {  				width: 141px;  				height: 203px;  				border: #ddd;  				background: white;  				left: 30px;  				top: 30px;  				bottom: 0px;  				position: fixed;  			}    			button {  				margin-right: 0px;  				float: right;  				margin-top: 0px;  			}    			/* 子框架样式 */  			.popup {  				position: absolute;  				top: 50%;  				left: 50%;  				transform: translate(-50%, -50%);  				padding: 20px;  				background-color: #00CED1;  				/* 设置背景色为青蓝色 */  				border: 5px solid #ccc;  				z-index: 999;  			}    			/* 链接样式 */  			.popup a {  				display: block;  				margin-bottom: 10px;  				color: #000;  				/* 设置链接文字颜色为黑色 */  				text-decoration: none;  			}  			#wrap{  			    height:auto;  			    padding-bottom:20px;  			}  			.shake{  			    width:128px;  			    height:128px;  			    border-radius:1.5rem;  			    overflow:hidden;  			}  			#footer{  			    position:fixed;  			    bottom:0px;  			      			}  			.bg{  			    height:100%;                  width:100%;                  background-repeat: no-repeat;                  background-size:cover;                  background-position: center;                  position:fixed;                  z-index:-10;                  background-image: url('../img/png/bg');  			}  			/* 隐藏垂直滚动条 */              body::-webkit-scrollbar {                width: 0;              }  		</style>  	</head>    	<body>  		<?php  	$lanip=$_SERVER["HTTP_HOST"];  ?>          <div class="bg"></div>  		<div id="wrap">  			<div id="top">  				<div id="logo">  					<img class="logo"  						src="">  					<h1>海纳思系统</h1>  					<h3>开源,免费,易用</h3>    					<div id="kg-btn" class="">  						<input class='tgl tgl-flip' id='qieh' type='checkbox' />  						<label class='tgl-btn' data-tg-off='外网' data-tg-on='局域网' for='qieh'></label>  					</div>  				</div>  			</div>  			<div id="main">  				<!-- 外网访问地址 -->  				<div id="app" class="app animated fadeInLeft">  					<ul>  						<li>  							<a href="/zhinan.html" target="_blank"><img class="shake" src="img/png/zhinan.png" /><strong>使用说明</strong></a>  						</li>  						<li>  							<a href="/phpinfo.php" target="_blank"><img class="shake"  									src="img/png/php.png" /><strong>设备信息</strong></a>  						</li>  						<li>  							<a href="#" onclick="openPopup()"><img class="shake"  									src="img/png/tools.png" /><strong>工具箱</strong></a>  						</li>  						<li>  							<a href="https://bbs.histb.com" target="_blank"><img class="shake"  									src="img/png/bbs.png" /><strong>互助社区</strong></a>  						</li>  						<?php                          $iconList = glob('icons_wan/*.html');  						foreach( $iconList as $item ){                              include($item);                          }  						?>    					</ul>  				</div>  				<div id="popup" class="popup" style="display: none;">  					<a href="/cal/" onclick="closePopup()" target="_blank"><img class="shake"  							src="img/png/calculator.png" /><strong>计算器</strong></a>  					<a href="/pw/" onclick="closePopup()" target="_blank"><img class="shake"  							src="img/png/pw.png" /><strong>随机密码生成器</strong></a>  					<a href="/sc/" onclick="closePopup()" target="_blank"><img class="shake"  							src="img/png/shoucangjia.png" /><strong>网址收藏夹</strong></a>  					<a href="#" onclick="closePopup()">关闭</a>  				</div>  				<!-- 内网网访问地址 -->  				<div id="app1" class="app animated fadeInRight">  					<ul>  						<li>  							<a href="/zhinan.html" target="_blank"><img class="shake" src="img/png/zhinan.png" /><strong>使用说明</strong></a>  						</li>  						<li>  							<a href="https://bbs.histb.com" target="_blank"><img class="shake"  									src="img/png/bbs.png" /><strong>互助社区</strong></a>  						</li>  						<?php                          $iconList = glob('icons_lan/*.html');  						foreach( $iconList as $item ){                              include($item);                          }  						?>  						<li>  							<a href="/controller.php" id="add"><img class="shake"  									src="img/png/add.png" /><strong>添加图标</strong></a>  						</li>  					</ul>  				</div>  				<div style="clear: both;"></div>  			</div>  		</div>  		<div id="footer">  			<div class="footer-contents">  				<div class="links">  					<div class="line">  						<a href="https://bbs.histb.com" target="_blank">海纳思交流论坛</a>&nbsp;&nbsp;&nbsp;&nbsp;<span  							class="footer-link-separator"></span>&nbsp;&nbsp;&nbsp;&nbsp;<span  							class="copyright">Copyright © 2023 </span>  					</div>  				</div>  			</div>  		</div>  		<form id="form" action="controller.php" style="display: none;" method="post" enctype="multipart/form-data">  			<input type="file" id="image" name="image">    		</form>  		<script src="js/jquery.min.js"></script>  		<script>  			$('button').click(function() {  				$('.adleft').hide()  			})  		</script>  		<script>  			// 打开子框架  			function openPopup() {  				var popup = document.getElementById("popup");  				popup.style.display = "block";  			}    			// 关闭子框架  			function closePopup() {  				var popup = document.getElementById("popup");  				popup.style.display = "none";  			}  			window.onload = function() {  			      				let diyList = document.getElementsByClassName("diy");  				for (let item of diyList) {  					let longPressTimeout;                        // 监听鼠标按下事件                        item.addEventListener('mousedown', (event) => {                            longPressTimeout = setTimeout(() => {                                console.log('长按事件触发');                                // 在这里添加你的长按事件处理代码                                event.preventDefault();      						let result = confirm("确定要删除吗?"); // 显示确认对话框      						if (result) {      							//执行删除操作      							let src = item.getAttribute("src").split("_");      							let html = src[0].replace("img/png/", "icons_") + "/" + src[1] + ".html";      							let png = item.getAttribute("src");                                  fetch('controller.php?filename=' + png)      								.catch(error => {      									console.error('Error:', error);      								});      							fetch('controller.php?filename=' + html)      								.catch(error => {      									console.error('Error:', error);      								});      							      							setTimeout(()=>{      							    location.reload();      							},1500);      						}                          }, 1000); // 这里设置的是1秒,你可以根据需要调整                        });                                                // 监听鼠标抬起事件                        item.addEventListener('mouseup', (event) => {                            clearTimeout(longPressTimeout);                        });  				}  				let add = document.getElementById("add");  				let longPressTimeout;                    // 监听鼠标按下事件                    add.addEventListener('mousedown', (event) => {                        longPressTimeout = setTimeout(() => {                            console.log('长按事件触发');                            // 在这里添加你的长按事件处理代码                            event.preventDefault();                          document.getElementById("image").click();  						document.getElementById('image').addEventListener('change', function(e) {                              var file = e.target.files[0];                              if (file) {                                var fileName = file.name; // 这就是文件名                                console.log(fileName); // 在控制台打印文件名                                document.getElementById("form").submit();                            }                            });                       }, 1000); // 这里设置的是1秒,你可以根据需要调整                    });                                        // 监听鼠标抬起事件                    add.addEventListener('mouseup', (event) => {                        clearTimeout(longPressTimeout);                    });  			}  		</script>  	</body>    </html>

controller.php

<!DOCTYPE html>  <html>  <head>      <link rel="stylesheet" type="text/css" href="css/bg.css" />      <style type="text/css">          body{              height: 100vh;              width:100vw;              font-size: 100%;              background-repeat: repeat-x;                background-size: auto 100%;               font-family: 'Helvetica Neue','Microsoft Yahei',SimHei,sans-serif;              /*color:#fff;*/              overflow:hidden;              background-image: url(../img/icloud.png);          }          .form {              position:fixed;              display: flex;  			flex-direction: column;              height:80%;  			width:80%;  			/*background-color: rgba(255, 255, 255, 0.5);  半透明背景 */    			background: linear-gradient(to bottom, rgba(255, 255, 255, 0.2), rgba(255, 255, 255, 0.8));  /*折射渐变 */  			/*box-shadow: 0px -10px 20px rgba(255, 255, 255, 0.5);  反射阴影 */  			box-shadow: 0rem 0rem 1rem rgba(255, 255, 255, 0.8); /* 高光阴影 */   			margin: 2% 8%;  			border-radius:1rem;  			padding:1%;  			font-weight: 400;              font-size: 2rem;              font-family: MyriadSetPro-Thin;              line-height: 2rem;  		}  		.form>div{  		    width:100%;  		    margin:2rem 0rem;  		    display:flex;  			align-items: center;  			flex-wrap: wrap;  		}  		.form-label{  		    height: 2rem;              text-align: center;  		}  		.radio{  		    width:2rem;  		    height:2rem  		}  		.form-value{  		    flex:1;              font-size: 2rem;              line-height: 3rem;  		    height: 3rem;  		    border-radius:.5rem;  		}  		.custom-file-input {              display: inline-block;              width: 100%; /* 或其他你想要的宽度 */              padding: 10px;              border-radius: 5px;              cursor: pointer;            }          .submit{              width:80%;              height:4rem;              font-size: 2rem;              line-height:4rem;              margin:auto 10%;          }          .bg{              left:0px;              top:0px;              height:100%;              width:100%;              background-repeat: no-repeat;              background-size:cover;              background-position: center;              position:fixed;              z-index:-10;              background-image: url('../img/png/bg');          }      </style>        </head>  <body>    <?php  // 检查表单是否已提交  if(isset($_POST["submit"])) {      // 检查文件是否成功上传      if(isset($_FILES["image"])) {          $file = $_FILES["image"];          // 获取文件的信息          $fileName = $file["name"];          $fileTmpPath = $file["tmp_name"];          $fileSize = $file["size"];          $fileError = $file["error"];          // 检查文件是否上传成功          if($fileError === 0) {              // 检查文件大小              if($fileSize <= 128 * 1024) { // 128 KB                  // 获取表单中的其他数据                  $name = $_POST["name"];                  $redirect = $_POST["redirect"];                  $position = $_POST["position"];                  $time = time();                  // 确定文件的存储路径                  $uploadsImgDirectory = "img/png/";                  $destinationPath = $uploadsImgDirectory . $position .'_'.$time;                    // 将文件移动到目标路径                  move_uploaded_file($fileTmpPath, $destinationPath);                    // 将表单数据写入本地文件                  $data = '<li>      <a href="'.(filter_var($redirect, FILTER_VALIDATE_URL)?'':'http://<?php echo $lanip ?>').$redirect.'" target="_blank"><img class="shake diy" src="img/png/'.$position .'_'.$time.'" /><strong>'.$name.'</strong></a>  </li>';                  $uploadsHtmlDirectory = "icons_$position/";                  $infoFilePath = $uploadsHtmlDirectory . $time.'.html';                  file_put_contents($infoFilePath, $data);                    echo "文件上传成功并数据写入成功!".(filter_var($redirect, FILTER_VALIDATE_URL)?'':'http://<?php echo $lanip ?>').$redirect;                  echo "<script>window.location.href='/index.html#'+new Date().getTime();</script>";              } else {                  echo "文件大小超过限制!";              }          } else {              echo "文件上传失败!";          }      }  }    //$filePath = '/'; // 设置要删除的目录路径    if(isset($_GET['filename'])) {      $filename = $_GET['filename'];      $fileToDelete = $filename;        if(is_file($fileToDelete)) {          if(unlink($fileToDelete)) {              echo "文件删除成功!";              // 可以执行其他操作,如日志记录等          } else {              echo "文件删除失败!";              // 可以执行其他操作,如日志记录等          }      } else {          echo "文件不存在!";          // 可以执行其他操作,如日志记录等      }  }  if(isset($_FILES['image'])) {      $file = $_FILES['image'];      echo $file;      $fileName = $file['name'];      $fileTmpPath = $file['tmp_name'];      $fileSize = $file['size'];      $fileError = $file['error'];      move_uploaded_file($fileTmpPath, "img/png/bg");      echo "<script>window.location.href='/index.html#'+new Date().getTime();</script>";  }  ?>  <div class="bg"></div>  <form class="form" method="post" enctype="multipart/form-data" onsubmit="return validateForm()">    <div>      <label class="form-label" for="name">名称:</label>      <input class="form-value" type="text" id="name" name="name">    </div>    <div>      <label class="form-label" for="name">位置:</label>      <input class="radio" type="radio" name="position" value="lan" checked>局域网       <input class="radio" type="radio" name="position" value="wan">外网     </div>    <div>      <label class="form-label" for="redirect">链接:</label>      <input class="form-value"type="text" id="redirect" name="redirect">    </div>                <div onclick="document.getElementById('image').click()">         <label class="form-label" for="image">图标:</label>        <span id="imageName">点击选择文件</span>          <input class="form-value" type="file" id="image" name="image" style="display: none;">        </div>    <br><br>      <input class="submit" type="submit" name="submit" value="提交">  </form>  <script>      function validateForm() {        // 获取表单元素        var nameInput = document.getElementById("name");        var redirectInput = document.getElementById("redirect");        var imageInput = document.getElementById("image");            // 检查名称是否为空        if (nameInput.value.trim() === "") {          alert("请填写名称!");          nameInput.focus();          return false;        }            // 检查跳转路径是否为空        if (redirectInput.value.trim() === "") {          alert("请填写链接!");          redirectInput.focus();          return false;        }            // 检查是否选择了图片        if (imageInput.files.length === 0) {          alert("请选择图片!");          imageInput.focus();          return false;        }        console.log(redirectInput.value);      // 表单验证通过,允许提交        return true;      }      document.getElementById('image').addEventListener('change', function(e) {          var file = e.target.files[0];          if (file) {            var fileName = file.name; // 这就是文件名            console.log(fileName); // 在控制台打印文件名            document.getElementById("imageName").innerHTML = fileName        }        });    </script>  </body>  </html>
 

About

搜索此博客