#!/bin/bash # wget -O tv.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/tv.sh && chmod +x tv.sh && ./tv.sh source common.sh apk_path="/tvhelper/apks/" # 定义红色文本 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 # 安装原生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 为空,使用默认值 version="1.0.6" fi echo $version } # 使用get_docker_version函数 docker_version=$(get_docker_version) 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 } # 函数用于检查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 connect_adb() { 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 echo -e "上次连接的设备是 ${GREEN}${last_name}${NC}IP地址为 ${GREEN}${last_ip}${NC}\n您是否要再次连接到此设备?确认请直接回车,否定输入n再回车[Y/n]" 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 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}" # 连接成功后,写入名称和IP地址到历史文件 echo "$(get_history_name)" >"$history_file" echo "${ip}" >>"$history_file" return 0 fi sleep 1 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 & sleep 2 # 给点时间让重启命令发出 disconnect_adb exit else echo "没有检测到已连接的设备。请先连接ADB" connect_adb fi } # 显示当前时区 show_timezone() { adb shell getprop persist.sys.timezone } #断开adb连接 disconnect_adb() { adb disconnect >/dev/null 2>&1 echo "ADB 已经断开" } # 添加主机名映射(解决安卓原生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 echo -e "仅支持英文字符和常规简单网址 不能支持 & * ? ,不建议重度使用此功能,重度使用请使用蓝牙键盘\n${YELLOW}如果输入clash订阅地址强烈建议使用第10项,${NC}\n ADB不适合处理特殊字符,且Openwrt下的adb版本也较低) \n输入【q】退出。输入【qk】删除20个字符。输入【blue】搜索蓝牙键盘。请您输入" read str 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}" elif [[ $str == "blue" ]]; then # 蓝牙 adb shell input keyevent KEYCODE_PAIRING echo -e "${YELLOW}已进入蓝牙配对模式。请在电视屏幕或显示器上根据提示配对您的蓝牙键盘${NC}" else 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 fi done else connect_adb fi } 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') echo "$ss" } # 安装apk install_apk() { local local_path=$1 local package_name=$2 local filename=$(basename "$local_path") 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=$! install_result=$(adb install -r $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 } # 批量安装apk功能 install_all_apks() { if check_adb_connected; then # 获取/tmp/upload目录下的apk文件列表 apk_files=($(ls /tvhelper/shells/data/*.apk 2>/dev/null)) total_files=${#apk_files[@]} # 检查是否有APK文件 if [ "$total_files" -eq "0" ]; then echo -e "${RED}/tvhelper/shells/data/ 目录下不包含任何apk文件,请先拷贝apk文件到此目录.${NC}" 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}" 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" } # 安装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}" install_apk "${apk_path}emotn.apk" "com.overseas.store.appstore" } # 安装当贝市场 install_dbmarket() { echo -e "${BLUE}安装过程若出现弹框,请点击详情后选择【仍然安装】即可${NC}" install_apk "${apk_path}dangbeimarket.apk" "com.dangbeimarket" } # 安装网络获取的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 } # 安装my-tv # release地址、包名、apk命名前缀 install_mytv_latest_apk() { echo -e "${BLUE}项目主页:https://github.com/lizongying/my-tv ${NC}" install_apk "${apk_path}mytv.apk" "com.lizongying.mytv" } # 安装bbll # release地址、包名、apk命名前缀 install_BBLL_latest_apk() { echo -e "${BLUE}项目主页:https://github.com/xiaye13579/BBLL ${NC}" install_apk "${apk_path}bbll.apk" "com.xx.blbl" } #根据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 } # 获取历史记录中盒子的名称 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 } # 获取电视盒子时区 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() { local xapk_local_path="${apk_path}mix.xapk" local xapkname=$(basename "$xapk_local_path") local extract_to="/tmp/mix/" mkdir -p "$extract_to" if unzip -o "$xapk_local_path" -d "$extract_to"; then 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() { wget -O kodi.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/kodi.sh && chmod +x kodi.sh && ./kodi.sh } # 安装fire tv版本youtube install_youtube_firetv() { echo -e "${BLUE}Fire TV版本Youtube无需谷歌框架 可用于所有安卓5.0以上电视盒子 ${NC}" 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 } # 进入tvbox安装助手 enter_tvbox_helper() { wget -O box.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/box.sh && chmod +x box.sh && ./box.sh } # 进入sony电视助手 enter_sonytv() { wget -O sony.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/sony.sh && chmod +x sony.sh && ./sony.sh } # 更新脚本 update_sh() { break echo "正在更新脚本..." # 下载最新的脚本到临时文件 wget -O /tmp/script.sh https://cafe.cpolar.cn/wkdaily/tvhelper-docker/raw/branch/master/shells/tv.sh # 替换当前脚本 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 } # 菜单 menu_options=( "连接ADB" "断开ADB" "安装Android原生TV必备精选Apps" "一键修改NTP(限原生TV,需重启)" "安装Play商店图标(仅google tv使用)" "自定义批量安装data目录下的所有apk" "替换系统桌面" "进入KODI助手" "进入TVBox安装助手" "进入Sony电视助手" "向TV端输入文字(限英文)" "显示Netflix影片码率" "模拟菜单键" "更新脚本" "赞助|打赏" ) commands=( ["连接ADB"]="connect_adb" ["断开ADB"]="disconnect_adb" ["安装Android原生TV必备精选Apps"]="android_tv_essentials" ["一键修改NTP(限原生TV,需重启)"]="modify_ntp" ["向TV端输入文字(限英文)"]="input_text" ["显示Netflix影片码率"]="show_nf_info" ["模拟菜单键"]="show_menu_keycode" ["安装Play商店图标(仅google tv使用)"]="show_playstore_icon" ["自定义批量安装data目录下的所有apk"]="install_all_apks" ["进入KODI助手"]="kodi_helper" ["进入TVBox安装助手"]="enter_tvbox_helper" ["进入Sony电视助手"]="enter_sonytv" ["更新脚本"]="update_sh" ["赞助|打赏"]="sponsor" ["替换系统桌面"]="replace_system_ui_menu" ) # 安装原生tv必备apps item_options=( "安装电视订阅助手" "安装Emotn Store应用商店" "安装当贝市场" "安装my-tv(lizongying)" "安装BBLL(xiaye13579)" "安装文件管理器+" "安装Downloader" "安装Mix-Apps用于显示全部应用" "返回主菜单" ) 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 else 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 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() { mkdir -p /tvhelper/shells/data clear echo "***********************************************************************" echo -e "* ${YELLOW}盒子助手Docker版 (v${docker_version})${NC} " echo -e "* ${GREEN}base Alpine Linux${NC} " 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