tvhelper-docker/shells/tv.sh

1062 lines
36 KiB
Bash
Raw Normal View History

2024-02-25 17:40:22 +08:00
#!/bin/bash
2024-08-04 16:04:23 +08:00
# wget -O tv.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/tv.sh && chmod +x tv.sh && ./tv.sh
2024-03-03 22:25:50 +08:00
source common.sh
apk_path="/tvhelper/apks/"
2024-02-25 17:40:22 +08:00
# 定义红色文本
RED='\033[0;31m'
# 无颜色
NC='\033[0m'
GREEN='\e[38;5;154m'
YELLOW="\e[93m"
BLUE="\e[96m"
# 菜单选项数组
declare -a menu_options
declare -A commands
2024-03-10 11:20:36 +08:00
# 安装原生tv必备菜单
declare -a item_options
declare -A commands_essentials
# 替换或恢复系统桌面
declare -a tv_model_options
declare -A tv_model_commands
get_docker_version() {
# 尝试从 /etc/environment 读取 APP_VERSION
if [ -f /etc/environment ]; then
source /etc/environment
fi
if [ -n "$APP_VERSION" ]; then
version=$APP_VERSION
else
# 若 /etc/environment 中的 APP_VERSION 为空,使用默认值
2024-08-04 16:35:25 +08:00
version="1.0.6"
2024-03-10 11:20:36 +08:00
fi
echo $version
}
# 使用get_docker_version函数
docker_version=$(get_docker_version)
2024-02-25 17:40:22 +08:00
show_user_tips() {
read -p "按 Enter 键继续..."
}
# 检查输入是否为整数
is_integer() {
if [[ $1 =~ ^-?[0-9]+$ ]]; then
return 0 # 0代表true/成功
else
return 1 # 非0代表false/失败
fi
}
# 判断adb是否连接成功
check_adb_connected() {
# 获取 adb devices 输出,跳过第一行(标题行),并检查每一行的状态
local connected_devices=$(adb devices | awk 'NR>1 {print $2}' | grep 'device$')
# 检查是否有设备已连接并且状态为 'device',即已授权
if [[ -n $connected_devices ]]; then
# ADB 已连接并且设备已授权
return 0
else
# ADB 设备未连接或未授权
return 1
fi
}
2024-03-03 22:25:50 +08:00
# 函数用于检查IP地址的合法性
is_valid_ip() {
if [[ $1 =~ ^[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
IFS='.' read -ra ip_parts <<<"$1"
for i in "${ip_parts[@]}"; do
if ((i < 0 || i > 255)); then
return 1
fi
done
return 0
else
return 1
fi
}
#连接adb并记录上次的ip
2024-02-25 17:40:22 +08:00
connect_adb() {
2024-03-03 22:25:50 +08:00
adb disconnect >/dev/null 2>&1
history_file="/tvhelper/shells/history"
if [[ -f "$history_file" ]]; then
last_ip=$(tail -n 1 "$history_file")
last_name=$(head -n 1 "$history_file")
# 检查历史中的IP地址是否合法
if is_valid_ip "$last_ip"; then
2024-03-10 11:20:36 +08:00
echo -e "上次连接的设备是 ${GREEN}${last_name}${NC}IP地址为 ${GREEN}${last_ip}${NC}\n您是否要再次连接到此设备?确认请直接回车,否定输入n再回车[Y/n]"
2024-03-03 22:25:50 +08:00
read answer
if [[ "$answer" == "N" || "$answer" == "n" ]]; then
echo -e "${YELLOW}请手动输入电视盒子的完整IP地址:${NC}"
read ip
else
ip=$last_ip
fi
else
echo -e "${RED}历史记录中的IP地址不合法,请手动输入电视盒子的完整IP地址:${NC}"
read ip
fi
else
echo -e "${YELLOW}请手动输入电视盒子的完整IP地址:${NC}"
read ip
fi
2024-02-25 17:40:22 +08:00
echo -e "${BLUE}首次使用,盒子上可能会提示授权弹框,给您半分钟时间来操作...【允许】${NC}"
adb connect ${ip}
for ((i = 1; i <= 30; i++)); do
echo -e "${YELLOW}${i}次尝试连接ADB,请在设备上点击【允许】按钮...${NC}"
device_status=$(adb devices | grep "${ip}:5555" | awk '{print $2}')
if [[ "$device_status" == "device" ]]; then
echo -e "${GREEN}ADB 已经连接成功啦,你可以放心操作了${NC}"
2024-03-03 22:25:50 +08:00
# 连接成功后写入名称和IP地址到历史文件
echo "$(get_history_name)" >"$history_file"
echo "${ip}" >>"$history_file"
2024-02-25 17:40:22 +08:00
return 0
fi
2024-03-03 22:25:50 +08:00
sleep 1
2024-02-25 17:40:22 +08:00
done
echo -e "${RED}连接超时,或者您点击了【取消】,请确认电视盒子的IP地址是否正确。如果问题持续存在,请检查设备的USB调试设置是否正确并重新连接adb${NC}"
}
# 一键修改NTP服务器地址
modify_ntp() {
echo -e "${BLUE}它的作用在于:解决安卓原生TV时间不正确和网络受限问题${NC}"
if check_adb_connected; then
adb shell settings put global ntp_server ntp3.aliyun.com
adb shell settings put global captive_portal_mode 1
adb shell settings put global captive_portal_detection_enabled 1
# 设置一个返回204 空内容的服务器
adb shell settings put global captive_portal_use_https 0
adb shell settings put global captive_portal_http_url http://connect.rom.miui.com/generate_204
echo -e "${GREEN}NTP服务器地址已经成功修改为国内,重启后请检查盒子的系统时间和时区${NC}"
echo -e "${RED}正在重启您的电视盒子或者电视机,请稍后.......${NC}"
# 5秒倒计时
for i in {5..1}; do
echo -e "${RED}$i${NC} 秒后将重启设备"
sleep 1
done
adb shell reboot &
2024-03-03 22:25:50 +08:00
sleep 2 # 给点时间让重启命令发出
disconnect_adb
exit
2024-02-25 17:40:22 +08:00
else
echo "没有检测到已连接的设备。请先连接ADB"
connect_adb
fi
}
# 显示当前时区
show_timezone() {
adb shell getprop persist.sys.timezone
}
#断开adb连接
disconnect_adb() {
2024-03-03 22:25:50 +08:00
adb disconnect >/dev/null 2>&1
2024-02-26 22:02:52 +08:00
echo "ADB 已经断开"
2024-02-25 17:40:22 +08:00
}
# 添加主机名映射(解决安卓原生TV首次连不上wifi的问题)
add_dhcp_domain() {
echo -e "${BLUE}它的作用在于:解决安卓原生TV首次使用连不上wifi的问题${NC}"
local domain_name="time.android.com"
local domain_ip="203.107.6.88"
# 检查是否存在相同的域名记录
existing_records=$(uci show dhcp | grep "dhcp.@domain\[[0-9]\+\].name='$domain_name'")
if [ -z "$existing_records" ]; then
# 添加新的域名记录
uci add dhcp domain
uci set "dhcp.@domain[-1].name=$domain_name"
uci set "dhcp.@domain[-1].ip=$domain_ip"
uci commit dhcp
echo
echo "已添加新的域名记录"
else
echo "相同的域名记录已存在,无需重复添加"
fi
echo -e "\n"
echo -e "time.android.com 203.107.6.88 "
}
show_nf_info() {
echo -e "${BLUE}播放Netflix影片的时候 屏幕左上角显示影片信息,再次执行则消失${NC}"
echo -e "${GREEN}Netflix INFO键已发送! 继续输入【m】模拟INFO键 或者输入q退出。${NC}"
if check_adb_connected; then
while true; do
read str
if [[ $str == "q" ]]; then
echo -e "${GREEN}退出输入模式。${NC}"
break # 当用户输入q时退出循环
elif [[ $str == "m" ]]; then
adb shell input keyevent KEYCODE_F8
echo -e "${GREEN}Netflix INFO键已发送! 继续输入【m】模拟INFO键 或者输入q退出。${NC}"
else
echo -e "${RED}请输入m或者输入q退出${NC}"
fi
done
else
connect_adb
fi
}
show_menu_keycode() {
echo -e "${BLUE}使用背景:${NC}\n${YELLOW}许多国产App还保留了菜单键的功能\n而原生TV盒子系统似乎逐渐放弃适配菜单键\n因此很多盒子附带的遥控器不会标配菜单键\n\n所以开发此功能,它会模拟触发菜单键\n请在盒子上观察是否有效,可反复执行${NC}"
echo -e "${GREEN}菜单键已发送! 继续输入字母【m】模拟菜单键 或者输入q退出。${NC}"
if check_adb_connected; then
while true; do
read str
if [[ $str == "q" ]]; then
echo -e "${GREEN}退出输入模式。${NC}"
break # 当用户输入q时退出循环
elif [[ $str == "m" ]]; then
adb shell input keyevent KEYCODE_MENU
echo -e "${GREEN}菜单键已发送! 继续输入m模拟菜单键 或者输入q退出。${NC}"
else
echo -e "${RED}请输入m或者输入q退出${NC}"
fi
done
else
connect_adb
fi
}
# 向电视盒子输入英文
input_text() {
echo -e "${BLUE}注意注意注意!请弹出键盘后再执行!每次输入会自动清空上次结果${NC}"
if check_adb_connected; then
while true; do
2024-03-03 22:25:50 +08:00
echo -e "仅支持英文字符和常规简单网址 不能支持 & * ? ,不建议重度使用此功能,重度使用请使用蓝牙键盘\n${YELLOW}如果输入clash订阅地址强烈建议使用第10项,${NC}\n ADB不适合处理特殊字符,且Openwrt下的adb版本也较低) \n输入【q】退出。输入【qk】删除20个字符。输入【blue】搜索蓝牙键盘。请您输入"
2024-02-25 17:40:22 +08:00
read str
2024-03-03 22:25:50 +08:00
2024-02-25 17:40:22 +08:00
if [[ $str == "q" ]]; then
echo -e "${GREEN}退出输入模式。${NC}"
break # 当用户输入q时退出循环
elif [[ $str == "qk" ]]; then
# 删除20个字符
for i in {1..20}; do
adb shell input keyevent KEYCODE_DEL
done
echo -e "${RED}哈哈!你可真够懒的!已帮你删除20个字符。继续输入或者输入q退出。${NC}"
2024-03-03 22:25:50 +08:00
elif [[ $str == "blue" ]]; then
# 蓝牙
adb shell input keyevent KEYCODE_PAIRING
echo -e "${YELLOW}已进入蓝牙配对模式。请在电视屏幕或显示器上根据提示配对您的蓝牙键盘${NC}"
2024-02-25 17:40:22 +08:00
else
2024-03-03 22:25:50 +08:00
after_str=$(convert_str "$str")
if adb shell input text "$after_str"; then
echo -e "${GREEN}[OK] 已发送! 继续输入或者输入q退出。${NC}"
else
# 如果adb命令失败提醒用户
echo -e "${RED}输入有误或adb命令执行失败请检查设备连接或输入的字符。${NC}"
fi
2024-02-25 17:40:22 +08:00
fi
done
else
connect_adb
fi
}
2024-03-03 22:25:50 +08:00
convert_str() {
local str="$1"
# 直接处理特殊字符,对于不确定的转义尝试去除反斜线
local ss=$(echo "$str" |
sed 's/[?]/\\\?/g' |
sed 's/[<]/\\</g' |
sed 's/[>]/\\>/g' |
sed 's/[|]/\\\|/g' |
sed 's/[~]/\\\~/g' |
sed 's/[\^]/\\\^/g' |
sed 's/ \$/$$/g' |
sed 's/ __/__ /g')
echo "$ss"
}
2024-02-25 17:40:22 +08:00
# 安装apk
install_apk() {
2024-03-03 22:25:50 +08:00
local local_path=$1
2024-02-25 17:40:22 +08:00
local package_name=$2
2024-03-03 22:25:50 +08:00
local filename=$(basename "$local_path")
2024-02-25 17:40:22 +08:00
if check_adb_connected; then
# 卸载旧版本的APK如果存在
adb uninstall "$package_name" >/dev/null 2>&1
2024-03-03 22:25:50 +08:00
echo -e "${GREEN}正在推送和安装${filename},请耐心等待...${NC}"
2024-02-25 17:40:22 +08:00
# 模拟安装进度
echo -ne "${BLUE}"
while true; do
echo -n ".."
sleep 1
done &
# 保存进度指示进程的PID
PROGRESS_PID=$!
2024-03-03 22:25:50 +08:00
install_result=$(adb install -r $local_path 2>&1)
2024-02-25 17:40:22 +08:00
# 安装完成后,终止进度指示进程
kill $PROGRESS_PID
wait $PROGRESS_PID 2>/dev/null
echo -e "${NC}\n"
# 检查安装结果
if [[ $install_result == *"Success"* ]]; then
echo -e "${GREEN}APK安装成功!请在盒子上查看${NC}"
else
echo -e "${RED}APK安装失败:$install_result${NC}"
fi
else
connect_adb
fi
}
# 批量安装apk功能
install_all_apks() {
if check_adb_connected; then
2024-03-03 22:25:50 +08:00
# 获取/tmp/upload目录下的apk文件列表
2024-02-27 12:25:43 +08:00
apk_files=($(ls /tvhelper/shells/data/*.apk 2>/dev/null))
2024-02-25 17:40:22 +08:00
total_files=${#apk_files[@]}
# 检查是否有APK文件
if [ "$total_files" -eq "0" ]; then
2024-03-03 22:25:50 +08:00
echo -e "${RED}/tvhelper/shells/data/ 目录下不包含任何apk文件,请先拷贝apk文件到此目录.${NC}"
2024-02-25 17:40:22 +08:00
return 1
fi
echo -e "${GREEN}================文件列表================${NC}"
for apk_file in "${apk_files[@]}"; do
filename=$(basename "$apk_file")
echo -e "${GREEN}$filename${NC}"
done
echo -e "${GREEN}========================================${NC}"
echo
echo -e "${BLUE}发现 $total_files 个APK. 开始安装...\n安装过程若出现弹框,请点击详情后选择【仍然安装】即可${NC}"
echo
# 安装APK文件并显示进度
for apk_file in "${apk_files[@]}"; do
filename=$(basename "$apk_file")
echo -ne "${YELLOW}Installing: $filename${NC} ${GREEN}"
echo
# 模拟安装进度
while true; do
echo -n ".."
sleep 1
done &
# 保存进度指示进程的PID
PROGRESS_PID=$!
# 执行实际的APK安装命令,并捕获输出
install_result=$(adb install -r "$apk_file" 2>&1)
# 安装完成后,终止进度指示进程
kill $PROGRESS_PID >/dev/null 2>&1
wait $PROGRESS_PID 2>/dev/null
echo -e "${NC}\nInstallation result: $install_result"
done
echo -e "${GREEN}所有APK安装完毕.${NC}"
else
connect_adb
fi
}
# 安装订阅助手
install_subhelper_apk() {
echo -e "${BLUE}电视订阅助手使用指南 前往观看:https://youtu.be/9NpYtPsJlGk ${NC}"
2024-03-03 22:25:50 +08:00
install_apk "${apk_path}subhelp14.apk" "com.wukongdaily.myclashsub"
}
# 安装play商店图标
show_playstore_icon() {
echo -e "${BLUE}这个apk仅用于google tv系统。因为google tv系统在首页并不会显示自家的谷歌商店图标${NC}"
install_apk "${apk_path}play-icon.apk" "com.android.vending.wk"
}
# 安装文件管理器
install_file_manager_plus() {
install_apk "${apk_path}File_Manager_Plus.apk" "com.alphainventor.filemanager"
}
# 安装Downloader
install_downloader() {
install_apk "${apk_path}downloader.apk" "com.esaba.downloader"
2024-02-25 17:40:22 +08:00
}
# 安装emotn store
install_emotn_store() {
echo -e "${BLUE}emotn_store使用指南1 前往观看:https://youtu.be/_S693NITNrs ${NC}"
echo -e "${YELLOW}emotn_store使用指南2 前往观看:https://youtu.be/lMhhIn4CQts ${NC}"
echo -e "${BLUE}安装过程若出现弹框,请点击详情后选择【仍然安装】即可${NC}"
2024-03-03 22:25:50 +08:00
install_apk "${apk_path}emotn.apk" "com.overseas.store.appstore"
2024-02-25 17:40:22 +08:00
}
# 安装当贝市场
install_dbmarket() {
echo -e "${BLUE}安装过程若出现弹框,请点击详情后选择【仍然安装】即可${NC}"
2024-03-03 22:25:50 +08:00
install_apk "${apk_path}dangbeimarket.apk" "com.dangbeimarket"
2024-02-25 17:40:22 +08:00
}
2024-03-03 22:25:50 +08:00
# 安装网络获取的apk
install_web_apk() {
local apk_download_url=$1
local package_name=$2
local filename=$(basename "$apk_download_url")
# 下载APK文件到临时目录
wget -O /tmp/$filename "$apk_download_url"
if check_adb_connected; then
# 卸载旧版本的APK如果存在
adb uninstall "$package_name" >/dev/null 2>&1
echo -e "${GREEN}正在推送和安装apk,请耐心等待...${NC}"
# 模拟安装进度
echo -ne "${BLUE}"
while true; do
echo -n ".."
sleep 1
done &
# 保存进度指示进程的PID
PROGRESS_PID=$!
install_result=$(adb install -r /tmp/$filename 2>&1)
# 安装完成后,终止进度指示进程
kill $PROGRESS_PID
wait $PROGRESS_PID 2>/dev/null
echo -e "${NC}\n"
# 检查安装结果
if [[ $install_result == *"Success"* ]]; then
echo -e "${GREEN}APK安装成功!请在盒子上查看${NC}"
else
echo -e "${RED}APK安装失败:$install_result${NC}"
fi
rm -rf /tmp/"$filename"
echo -e "${YELLOW}临时文件/tmp/${filename}已清理${NC}"
else
connect_adb
fi
2024-02-25 17:40:22 +08:00
}
# 安装my-tv
# release地址、包名、apk命名前缀
install_mytv_latest_apk() {
echo -e "${BLUE}项目主页:https://github.com/lizongying/my-tv ${NC}"
2024-03-03 22:25:50 +08:00
install_apk "${apk_path}mytv.apk" "com.lizongying.mytv"
2024-02-25 17:40:22 +08:00
}
# 安装bbll
# release地址、包名、apk命名前缀
install_BBLL_latest_apk() {
echo -e "${BLUE}项目主页:https://github.com/xiaye13579/BBLL ${NC}"
2024-03-03 22:25:50 +08:00
install_apk "${apk_path}bbll.apk" "com.xx.blbl"
2024-02-25 17:40:22 +08:00
}
#根据apk地址和包名 安装apk
install_apk_by_url() {
local releases_url=$1
local package_name=$2
local name_prefix=$3
# 使用get_apk_url函数获取APK的下载链接
local apk_url=$(get_apk_url_by_name_prefix "$releases_url" "$name_prefix")
if [ -z "$apk_url" ]; then
echo "APK download URL could not be found."
return 1
fi
# 从URL中提取文件名
local filename=$(basename "$apk_url")
echo -e "${YELLOW}已获取最新版下载地址:\n$apk_url${NC}"
# 使用curl下载APK文件并保存到/tmp目录
echo -e "${GREEN}Downloading APK to /tmp/$filename ... ${NC}"
curl -L "$apk_url" -o /tmp/"$filename"
if [ $? -eq 0 ]; then
echo -e "${GREEN}APK downloaded successfully to /tmp/$filename ${NC}"
if check_adb_connected; then
# 卸载旧版本的APK
adb uninstall "$package_name" >/dev/null 2>&1
echo -e "${GREEN}正在推送和安装$filename 请耐心等待...${NC}"
# 模拟安装进度
echo -ne "${BLUE}"
while true; do
echo -n ".."
sleep 1
done &
# 保存进度指示进程的PID
PROGRESS_PID=$!
# 安装新版本的APK
install_result=$(adb install /tmp/"$filename" 2>&1)
# 安装完成后,终止进度指示进程
kill $PROGRESS_PID
wait $PROGRESS_PID 2>/dev/null
echo -e "${NC}\n"
# 检查安装结果
if [[ $install_result == *"Success"* ]]; then
echo -e "${GREEN}APK安装成功!请在盒子上查看${NC}"
else
echo -e "${RED}APK安装失败:$install_result${NC}"
fi
rm -rf /tmp/"$filename"
echo -e "${YELLOW}临时文件/tmp/${filename}已清理${NC}"
else
connect_adb
fi
else
echo "Failed to download APK."
return 1
fi
}
#根据release地址和命名前缀获取apk地址
get_apk_url_by_name_prefix() {
if [ $# -eq 0 ]; then
echo "需要提供GitHub releases页面的URL作为参数。"
return 1
fi
local releases_url=$1
local name_prefix=$2
# 使用curl获取重定向的URL
latest_url=$(curl -Ls -o /dev/null -w "%{url_effective}" "$releases_url")
# 使用sed从URL中提取tag值,并保留前导字符'v'
tag=$(echo $latest_url | sed 's|.*/v|v|')
# 检查是否成功获取到tag
if [ -z "$tag" ]; then
echo "未找到最新的release tag。"
return 1
fi
# 拼接APK下载链接
local repo_path=$(echo "$releases_url" | sed -n 's|https://github.com/\(.*\)/releases/latest|\1|p')
apk_download_url="https://github.com/${repo_path}/releases/download/${tag}/${name_prefix}${tag}.apk"
echo "$apk_download_url"
}
get_status() {
if check_adb_connected; then
adb_status="${GREEN}已连接且已授权${NC}"
else
adb_status="${RED}未连接${NC}"
fi
echo -e "* 与电视盒子的连接状态:$adb_status"
}
# 获取电视盒子型号
get_tvbox_model_name() {
if check_adb_connected; then
# 获取设备型号
local model=$(adb shell getprop ro.product.model)
# 获取设备制造商
local manufacturer=$(adb shell getprop ro.product.manufacturer)
# 清除换行符
model=$(echo $model | tr -d '\r' | tr -d '\n')
manufacturer=$(echo $manufacturer | tr -d '\r' | tr -d '\n')
echo -e "* 当前电视盒子型号:${BLUE}$manufacturer $model${NC}"
else
echo -e "* 当前电视盒子型号:${BLUE}请先连接ADB${NC}"
fi
}
2024-03-03 22:25:50 +08:00
# 获取历史记录中盒子的名称
get_history_name() {
if check_adb_connected; then
# 获取设备型号
local model=$(adb shell getprop ro.product.model)
# 获取设备制造商
local manufacturer=$(adb shell getprop ro.product.manufacturer)
# 清除换行符
model=$(echo $model | tr -d '\r' | tr -d '\n')
manufacturer=$(echo $manufacturer | tr -d '\r' | tr -d '\n')
echo "$manufacturer $model "
else
echo -e ""
fi
}
2024-02-25 17:40:22 +08:00
# 获取电视盒子时区
get_tvbox_timezone() {
if check_adb_connected; then
# 获取设备时区
device_timezone=$(adb shell getprop persist.sys.timezone)
# 获取设备系统时间,格式化为“年月日 时:分”
device_time=$(adb shell date "+%Y年%m月%d日 %H:%M")
echo -e "* 当前电视盒子时区:${YELLOW}$device_timezone${NC}"
echo -e "* 当前电视盒子时间:${YELLOW}$device_time${NC}"
else
echo -e "* 当前电视盒子时区:${BLUE}请先连接ADB${NC}"
echo -e "* 当前电视盒子时间:${BLUE}请先连接ADB${NC}"
fi
}
# 安装mix apps 用于显示全部app
install_mixapps() {
2024-03-03 22:25:50 +08:00
local xapk_local_path="${apk_path}mix.xapk"
local xapkname=$(basename "$xapk_local_path")
2024-02-25 17:40:22 +08:00
local extract_to="/tmp/mix/"
mkdir -p "$extract_to"
2024-03-03 22:25:50 +08:00
if unzip -o "$xapk_local_path" -d "$extract_to"; then
2024-02-25 17:40:22 +08:00
echo "XAPK文件解压成功,准备安装..."
else
echo "XAPK文件解压失败,请检查文件是否损坏或尝试重新下载。"
return 1 # 返回一个错误状态
fi
apk_files=$(find "$extract_to" -type f -name "*.apk")
echo -e "解压后的多个apk:\n$apk_files"
echo -ne "${YELLOW}正在安装: $xapkname${NC} ${GREEN}"
echo
# 模拟安装进度
while true; do
echo -n ".."
sleep 1
done &
# 保存进度指示进程的PID
PROGRESS_PID=$!
# 执行实际的APK安装命令,并捕获输出
install_result=$(adb install-multiple $apk_files 2>&1)
# 安装完成后,终止进度指示进程
kill $PROGRESS_PID >/dev/null 2>&1
wait $PROGRESS_PID 2>/dev/null
echo -e "${NC}\nInstallation result: $install_result"
if [ $? -eq 0 ]; then
echo -e "${GREEN} 安装成功 ${NC}"
# 安装成功后删除解压的文件和原始XAPK文件
echo -e "${RED}正在删除临时文件...${NC}"
rm -rf "$extract_to" # 删除解压目录
echo -e "${GREEN}临时文件删除完成,行啦,在盒子上查看吧!${NC}"
else
echo -e "${RED}安装失败${NC}"
fi
}
# 进入KODI助手
kodi_helper() {
2024-08-04 16:04:23 +08:00
wget -O kodi.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/kodi.sh && chmod +x kodi.sh && ./kodi.sh
2024-02-25 17:40:22 +08:00
}
# 安装fire tv版本youtube
install_youtube_firetv() {
echo -e "${BLUE}Fire TV版本Youtube无需谷歌框架 可用于所有安卓5.0以上电视盒子 ${NC}"
2024-03-03 22:25:50 +08:00
local apk_local_path="/tvhelper/apks/youtube.apk"
if check_adb_connected; then
echo -e "${GREEN}正在推送和安装fire tv版youtube,请耐心等待...${NC}"
# 模拟安装进度
echo -ne "${BLUE}"
while true; do
echo -n ".."
sleep 1
done &
# 保存进度指示进程的PID
PROGRESS_PID=$!
install_result=$(adb install -r $apk_local_path 2>&1)
# 安装完成后,终止进度指示进程
kill $PROGRESS_PID
wait $PROGRESS_PID 2>/dev/null
echo -e "${NC}\n"
# 检查安装结果
if [[ $install_result == *"Success"* ]]; then
echo -e "${GREEN}APK安装成功!请在盒子上查看${NC}"
else
echo -e "${RED}APK安装失败:$install_result${NC}"
fi
else
connect_adb
fi
2024-02-25 17:40:22 +08:00
}
# 进入tvbox安装助手
enter_tvbox_helper() {
2024-08-04 16:04:23 +08:00
wget -O box.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/box.sh && chmod +x box.sh && ./box.sh
2024-02-25 17:40:22 +08:00
}
# 进入sony电视助手
enter_sonytv() {
2024-08-04 16:04:23 +08:00
wget -O sony.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/sony.sh && chmod +x sony.sh && ./sony.sh
2024-02-25 17:40:22 +08:00
}
2024-03-10 11:20:36 +08:00
# 更新脚本
update_sh() {
break
echo "正在更新脚本..."
# 下载最新的脚本到临时文件
2024-08-04 16:04:23 +08:00
wget -O /tmp/script.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/tv.sh
2024-03-10 11:20:36 +08:00
# 替换当前脚本
if [ -f /tmp/script.sh ]; then
chmod +x /tmp/script.sh
cp /tmp/script.sh /tvhelper/shells/tv.sh
echo "脚本更新成功。即将重新启动脚本。"
# 使用 exec 来重新启动脚本,替换当前进程
exec /tvhelper/shells/tv.sh
else
echo "更新失败。"
fi
}
2024-03-03 22:25:50 +08:00
2024-02-25 17:40:22 +08:00
# 菜单
menu_options=(
"连接ADB"
"断开ADB"
2024-03-10 11:20:36 +08:00
"安装Android原生TV必备精选Apps"
"一键修改NTP(限原生TV,需重启)"
"安装Play商店图标(仅google tv使用)"
2024-02-27 12:25:43 +08:00
"自定义批量安装data目录下的所有apk"
2024-03-10 11:20:36 +08:00
"替换系统桌面"
2024-02-25 17:40:22 +08:00
"进入KODI助手"
"进入TVBox安装助手"
"进入Sony电视助手"
2024-03-10 11:20:36 +08:00
"向TV端输入文字(限英文)"
"显示Netflix影片码率"
"模拟菜单键"
2024-02-25 17:40:22 +08:00
"更新脚本"
"赞助|打赏"
)
commands=(
["连接ADB"]="connect_adb"
["断开ADB"]="disconnect_adb"
2024-03-10 11:20:36 +08:00
["安装Android原生TV必备精选Apps"]="android_tv_essentials"
["一键修改NTP(限原生TV,需重启)"]="modify_ntp"
2024-02-25 17:40:22 +08:00
["向TV端输入文字(限英文)"]="input_text"
["显示Netflix影片码率"]="show_nf_info"
["模拟菜单键"]="show_menu_keycode"
2024-03-10 11:20:36 +08:00
["安装Play商店图标(仅google tv使用)"]="show_playstore_icon"
2024-02-27 12:25:43 +08:00
["自定义批量安装data目录下的所有apk"]="install_all_apks"
2024-02-25 17:40:22 +08:00
["进入KODI助手"]="kodi_helper"
["进入TVBox安装助手"]="enter_tvbox_helper"
["进入Sony电视助手"]="enter_sonytv"
["更新脚本"]="update_sh"
2024-03-10 11:20:36 +08:00
["赞助|打赏"]="sponsor"
["替换系统桌面"]="replace_system_ui_menu"
)
# 安装原生tv必备apps
item_options=(
"安装电视订阅助手"
"安装Emotn Store应用商店"
"安装当贝市场"
"安装my-tv(lizongying)"
"安装BBLL(xiaye13579)"
"安装文件管理器+"
"安装Downloader"
"安装Mix-Apps用于显示全部应用"
"返回主菜单"
2024-02-25 17:40:22 +08:00
)
2024-03-10 11:20:36 +08:00
commands_essentials=(
["安装电视订阅助手"]="install_subhelper_apk"
["安装Emotn Store应用商店"]="install_emotn_store"
["安装当贝市场"]="install_dbmarket"
["安装my-tv(lizongying)"]="install_mytv_latest_apk"
["安装BBLL(xiaye13579)"]="install_BBLL_latest_apk"
["安装文件管理器+"]="install_file_manager_plus"
["安装Downloader"]="install_downloader"
["安装Mix-Apps用于显示全部应用"]="install_mixapps"
)
# 替换或恢复系统桌面
tv_model_options=(
"替换/恢复 索尼Sony电视系统桌面"
"替换/恢复 小米(盒子/电视)系统桌面"
"替换/恢复 小米盒子国际版系统桌面"
"替换/恢复 GoogleTV系统桌面"
"替换/恢复 安卓原生TV系统桌面(原生类型TV通用)"
"返回主菜单"
)
tv_model_commands=(
["替换/恢复 索尼Sony电视系统桌面"]="replace_sony_ui"
["替换/恢复 小米(盒子/电视)系统桌面"]="replace_xiaomi_ui"
["替换/恢复 小米盒子国际版系统桌面"]="replace_xiaomi_global_ui"
["替换/恢复 GoogleTV系统桌面"]="toggle_googletv_system_ui"
["替换/恢复 安卓原生TV系统桌面(原生类型TV通用)"]="replace_normal_androidtv_ui"
)
# 定义安卓原生TV必备子菜单函数
android_tv_essentials() {
while true; do
echo -e "${GREEN}原生TV必备精选Apps:${NC}"
for i in "${!item_options[@]}"; do
echo " ($((i + 1))) ${item_options[$i]}"
done
echo "请选择一个选项,或按q返回主菜单:"
read -r choice
# 检查输入是否为退出命令
if [[ "$choice" == "q" ]]; then
break
fi
# 检查输入是否为数字
if ! [[ $choice =~ ^[0-9]+$ ]]; then
echo -e " ${RED}请输入有效数字!${NC}"
continue
fi
# 检查数字是否在有效范围内
if [[ $choice -lt 1 ]] || [[ $choice -gt ${#item_options[@]} ]]; then
echo -e " ${RED}选项超出范围!${NC}"
echo -e " ${YELLOW}请输入 1 到 ${#item_options[@]} 之间的数字。${NC}"
continue
fi
# 处理返回主菜单
if [[ $choice -eq ${#item_options[@]} ]]; then
break
fi
local selected_option="${item_options[$((choice - 1))]}"
command_item_run="${commands_essentials["$selected_option"]}"
# 检查是否存在对应的命令并执行
if [ -z "$command_item_run" ]; then
echo -e " ${RED}无效选项,请重新选择。${NC}"
else
eval "$command_item_run"
fi
done
}
# 根据品牌替换系统桌面
replace_system_ui_menu() {
local apk_path="/tvhelper/apks/ui.apk"
# 检查APK文件是否存在
if [ ! -f "$apk_path" ]; then
echo -e "${RED}错误: 要替换的桌面APK文件不存在,请更新docker镜像后重试。${NC}"
return 1
fi
while true; do
echo -e "${GREEN}目前支持替换桌面的电视盒子或电视品牌如下:${NC}"
for i in "${!tv_model_options[@]}"; do
echo " ($((i + 1))) ${tv_model_options[$i]}"
done
echo "请选择一个选项,或按q返回主菜单:"
read -r choice
# 检查输入是否为退出命令
if [[ "$choice" == "q" ]]; then
break
fi
# 检查输入是否为数字
if ! [[ $choice =~ ^[0-9]+$ ]]; then
echo -e " ${RED}请输入有效数字!${NC}"
continue
fi
# 检查数字是否在有效范围内
if [[ $choice -lt 1 ]] || [[ $choice -gt ${#tv_model_options[@]} ]]; then
echo -e " ${RED}选项超出范围!${NC}"
echo -e " ${YELLOW}请输入 1 到 ${#tv_model_options[@]} 之间的数字。${NC}"
continue
fi
# 处理返回主菜单
if [[ $choice -eq ${#tv_model_options[@]} ]]; then
break
fi
local selected_option="${tv_model_options[$((choice - 1))]}"
local command_item_run="${tv_model_commands["$selected_option"]}"
# 检查是否存在对应的命令并执行
if [ -z "$command_item_run" ]; then
echo -e " ${RED}无效选项,请重新选择。${NC}"
else
eval "$command_item_run"
fi
done
}
replace_xiaomi_ui() {
local system_ui_package="com.mitv.tvhome"
toggle_system_ui "${system_ui_package}"
}
replace_xiaomi_global_ui() {
local system_ui_package="com.google.android.tvlauncher"
toggle_system_ui "${system_ui_package}"
}
replace_sony_ui() {
local system_ui_package="com.dangbei.TVHomeLauncher"
toggle_system_ui "${system_ui_package}"
}
replace_xiaomi_global_ui() {
replace_normal_androidtv_ui
}
replace_normal_androidtv_ui() {
local system_ui_package="com.google.android.tvlauncher"
toggle_system_ui "${system_ui_package}"
}
check_emotnui_installed(){
local package_name="com.oversea.aslauncher"
local apk_path="/tvhelper/apks/ui.apk"
# 检查APK文件是否存在
if [ ! -f "$apk_path" ]; then
echo -e "${RED}错误: APK文件不存在,请更新docker镜像后重试,确保docker镜像版本 >= 1.0.3${NC}"
return 1
fi
# 检查 com.oversea.aslauncher 是否已安装
if ! adb shell pm list packages | grep -q "$package_name"; then
echo -e "${GREEN}EmotnUI 未安装,开始安装...请稍后${NC}"
# 安装 com.oversea.aslauncher 应用
if adb install -r "$apk_path" >/dev/null 2>&1; then
echo -e "${GREEN}第三方桌面安装成功${NC}"
else
echo -e "${RED}应用安装失败,请检查APK文件路径和设备连接状态。若apk不存在请更新docker镜像。${NC}"
return
fi
2024-02-25 17:40:22 +08:00
else
2024-03-10 11:20:36 +08:00
echo -e "${GREEN}第三方桌面EmotnUI已安装。${NC}"
fi
}
toggle_googletv_system_ui() {
local system_ui_package="com.google.android.apps.tv.launcherx"
local system_setup_package="com.google.android.tungsten.setupwraith"
#判断emotnui是否安装
check_emotnui_installed
# 检查系统桌面是否已被禁用
if adb shell pm list packages -d | grep -q "$system_ui_package"; then
# 若已被禁用,则启用系统桌面
if adb shell pm enable "$system_ui_package" >/dev/null 2>&1 && adb shell pm enable "$system_setup_package" >/dev/null 2>&1; then
echo -e "${GREEN}恭喜您,您的系统桌面又回来啦! 请按HOME键确认。${NC}"
adb shell input keyevent KEYCODE_HOME
else
echo -e "${RED}启用系统桌面或其他应用失败,请检查设备连接状态和权限。${NC}"
fi
else
# 若未被禁用,则禁用系统桌面
if adb shell pm disable-user --user 0 "$system_ui_package" >/dev/null 2>&1 &&
adb shell pm disable-user --user 0 "$system_setup_package" >/dev/null 2>&1; then
echo -e "${GREEN}恭喜您,新桌面替换成功。点击HOME键 查看新桌面哦。${NC}"
adb shell input keyevent KEYCODE_HOME
else
echo -e "${RED}禁用系统桌面失败,请检查设备连接状态和权限。${NC}"
fi
fi
}
# 替换或恢复系统桌面
toggle_system_ui() {
local system_ui_package=$1
#判断emotnui是否安装
check_emotnui_installed
# 检查系统桌面是否已被禁用
if adb shell pm list packages -d | grep -q "$system_ui_package"; then
# 若已被禁用,则启用系统桌面
if adb shell pm enable "$system_ui_package" >/dev/null 2>&1; then
echo -e "${GREEN}恭喜您,您的系统桌面又回来啦! 请按HOME键确认。${NC}"
adb shell input keyevent KEYCODE_HOME
else
echo -e "${RED}启用系统桌面失败,请检查设备连接状态和权限。${NC}"
fi
else
# 若未被禁用,则禁用系统桌面
if adb shell pm disable-user --user 0 "$system_ui_package" >/dev/null 2>&1; then
echo -e "${GREEN}恭喜您,新桌面替换成功。点击HOME键 查看新桌面哦。${NC}"
adb shell input keyevent KEYCODE_HOME
else
echo -e "${RED}禁用系统桌面失败,请检查设备连接状态和权限。${NC}"
fi
2024-02-25 17:40:22 +08:00
fi
}
# 处理菜单
handle_choice() {
local choice=$1
# 检查输入是否为空
if [[ -z $choice ]]; then
echo -e "${RED}输入不能为空,请重新选择。${NC}"
return
fi
# 检查输入是否为数字
if ! [[ $choice =~ ^[0-9]+$ ]]; then
echo -e "${RED}请输入有效数字!${NC}"
return
fi
# 检查数字是否在有效范围内
if [[ $choice -lt 1 ]] || [[ $choice -gt ${#menu_options[@]} ]]; then
echo -e "${RED}选项超出范围!${NC}"
echo -e "${YELLOW}请输入 1 到 ${#menu_options[@]} 之间的数字。${NC}"
return
fi
local selected_option="${menu_options[$choice - 1]}"
local command_to_run="${commands[$selected_option]}"
# 检查是否存在对应的命令
if [ -z "$command_to_run" ]; then
echo -e "${RED}无效选项,请重新选择。${NC}"
return
fi
# 使用eval执行命令
eval "$command_to_run"
}
show_menu() {
2024-02-27 12:25:43 +08:00
mkdir -p /tvhelper/shells/data
2024-02-25 17:40:22 +08:00
clear
echo "***********************************************************************"
2024-03-10 11:20:36 +08:00
echo -e "* ${YELLOW}盒子助手Docker版 (v${docker_version})${NC} "
2024-03-03 22:25:50 +08:00
echo -e "* ${GREEN}base Alpine Linux${NC} "
2024-02-25 17:40:22 +08:00
echo -e "* ${RED}请确保电视盒子和Docker宿主机处于${NC}${BLUE}同一网段${NC}\n* ${RED}且电视盒子开启了${NC}${BLUE}USB调试模式(adb开关)${NC} "
echo "**********************************************************************"
echo "$(get_status)"
echo "$(get_tvbox_model_name)"
echo "$(get_tvbox_timezone)"
echo "**********************************************************************"
echo "请选择操作:"
for i in "${!menu_options[@]}"; do
echo -e "${BLUE}$((i + 1)). ${menu_options[i]}${NC}"
done
}
while true; do
show_menu
read -p "请输入选项的序号(输入q退出): " choice
if [[ $choice == 'q' ]]; then
disconnect_adb
break
fi
handle_choice $choice
echo "按任意键继续..."
read -n 1 # 等待用户按键
done