#!/bin/bash # wget -O tv.sh https://raw.githubusercontent.com/wukongdaily/tvhelper/master/shells/tv.sh && chmod +x tv.sh && ./tv.sh #判断是否为x86软路由 proxy="" if [ $# -gt 0 ]; then proxy="https://mirror.ghproxy.com/" fi is_x86_64_router() { DISTRIB_ARCH=$(cat /etc/openwrt_release | grep "DISTRIB_ARCH" | cut -d "'" -f 2) if [ "$DISTRIB_ARCH" = "x86_64" ]; then return 0 else return 1 fi } download_common_shell() { if [ ! -f common.sh ]; then wget -O common.sh "${proxy}https://raw.githubusercontent.com/wukongdaily/tvhelper/master/shells/common.sh" && chmod +x common.sh fi source common.sh } #******************************************************** # 定义红色文本 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 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_installed() { if opkg list-installed | grep -q "^adb "; then return 0 # 表示 adb 已安装 else return 1 # 表示 adb 未安装 fi } # 判断adb是否连接成功 check_adb_connected() { if check_adb_installed; then # 获取 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 else # 表示 adb 未安装 return 1 fi } # 安装adb工具 install_adb() { echo -e "${BLUE}绝大多数软路由自带ADB 只有少数OpenWrt硬路由才需要安装ADB${NC}" # 调用函数并根据返回值判断 if check_adb_installed; then echo -e "${YELLOW}您的路由器已经安装了ADB工具${NC}" else opkg update echo -e "${YELLOW}正在尝试安装adb${NC}" if opkg install adb; then echo -e "${GREEN}adb 安装成功!${NC}" else echo -e "${RED}adb 安装失败,请检查日志以获取更多信息。${NC}" fi fi } # 连接adb connect_adb() { install_adb # 尝试自动获取网关地址 #gateway_ip=$(ip route show default | grep default | awk '{print $3}') gateway_ip=$(ip a show br-lan | grep 'inet ' | awk '{print $2}' | cut -d'/' -f1) if [ -z "$gateway_ip" ]; then echo -e "${RED}无法自动获取网关IP地址,请手动输入电视盒子的完整IP地址:${NC}" read ip else # 提取网关IP地址的前缀 gateway_prefix=$(echo $gateway_ip | sed 's/\.[0-9]*$//'). echo -e "${YELLOW}请输入电视盒子的ip地址(${NC}${BLUE}${gateway_prefix}${NC}${YELLOW})的最后一段数字${NC}" read end_number if is_integer "$end_number"; then # 使用动态获取的网关前缀 ip=${gateway_prefix}${end_number} else echo -e "${RED}错误: 请输入整数。${NC}" return 1 fi fi adb disconnect 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}" return 0 fi sleep 1 # 每次检测间隔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() { if check_adb_installed; then adb disconnect >/dev/null 2>&1 echo "ADB 已经断开" else echo -e "${YELLOW}您还没有安装ADB${NC}" fi } # 添加主机名映射(解决安卓原生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 apk_download_url=$1 local package_name=$2 local filename=$(basename "$apk_download_url") # 下载APK文件到临时目录 wget -O /tmp/$filename "${proxy}$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 } # 批量安装apk功能 install_all_apks() { if check_adb_connected; then # 获取/tmp/upload目录下的apk文件列表 apk_files=($(ls /tmp/upload/*.apk 2>/dev/null)) total_files=${#apk_files[@]} # 检查是否有APK文件 if [ "$total_files" -eq "0" ]; then echo -e "${RED}/tmp/upload 目录下不包含任何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 "https://github.com/wukongdaily/tvhelper/raw/master/apks/subhelp14.apk" "com.wukongdaily.myclashsub" } # 安装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 "https://app.keeflys.com/20220107/com.overseas.store.appstore_1.0.40_a973.apk" "com.overseas.store.appstore" } # 安装当贝市场 install_dbmarket() { echo -e "${BLUE}安装过程若出现弹框,请点击详情后选择【仍然安装】即可${NC}" install_apk "https://webapk.dangbei.net/update/dangbeimarket.apk" "com.dangbeimarket" } # 安装play商店图标 show_playstore_icon() { echo -e "${BLUE}这个apk仅用于google tv系统。因为google tv系统在首页并不会显示自家的谷歌商店图标${NC}" install_apk "https://github.com/wukongdaily/tvhelper/raw/master/apks/play-icon.apk" "com.android.vending.wk" } # 安装my-tv # release地址、包名、apk命名前缀 install_mytv_latest_apk() { echo -e "${BLUE}项目主页:https://github.com/lizongying/my-tv ${NC}" install_apk_by_url "https://github.com/lizongying/my-tv/releases/latest" "com.lizongying.mytv" "my-tv-" } # 安装bbll # release地址、包名、apk命名前缀 install_BBLL_latest_apk() { echo -e "${BLUE}项目主页:https://github.com/xiaye13579/BBLL ${NC}" install_apk_by_url "https://github.com/xiaye13579/BBLL/releases/latest" "com.xx.blbl" "BLBL_release_" } # 安装文件管理器 install_file_manager_plus() { install_apk "https://github.com/wukongdaily/tvhelper/raw/master/apks/File_Manager_Plus.apk" "com.alphainventor.filemanager" } # 安装Downloader install_downloader() { install_apk "https://github.com/wukongdaily/tvhelper/raw/master/apks/downloader.apk" "com.esaba.downloader" } #根据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" } # 添加emotn域名 add_emotn_domain() { # 检查 passwall 的代理域名文件是否存在 if [ -f "/usr/share/passwall/rules/proxy_host" ]; then sed -i "s/keeflys.com//g" "/usr/share/passwall/rules/proxy_host" echo -n "keeflys.com" | tee -a /usr/share/passwall/rules/proxy_host echo -e "${GREEN}在pw代理域名中 添加成功!${NC}" else echo -e "${RED}在passwall代理域名中 添加失败! 请确保 passwall 已安装${NC}" fi # 检查 SSRP 的黑名单文件是否存在 if [ -f "/etc/ssrplus/black.list" ]; then sed -i "s/keeflys.com//g" "/etc/ssrplus/black.list" echo -n "keeflys.com" | tee -a /etc/ssrplus/black.list echo -e "${GREEN}在SSRP代理域名中 添加成功!${NC}" else echo -e "${RED}添加失败! 请确保 SSRP 已安装${NC}" fi echo -e "\n\n" } 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_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 } # 能否访问Github check_github_connected() { # Ping GitHub域名并提取时间 ping_time=$(ping -c 1 raw.githubusercontent.com | grep 'time=' | awk -F'time=' '{print $2}' | awk '{print $1}') if [ -n "$ping_time" ]; then echo -e "* 当前路由器访问Github延时:${BLUE}${ping_time}ms${NC}" else echo -e "* 当前路由器访问Github延时:${RED}超时${NC}" fi } ##获取软路由型号信息 get_router_name() { if is_x86_64_router; then model_name=$(grep "model name" /proc/cpuinfo | head -n 1 | awk -F: '{print $2}' | sed 's/^[ \t]*//;s/[ \t]*$//') echo "$model_name" else model_info=$(cat /tmp/sysinfo/model) echo "$model_info" fi } # 安装mix apps 用于显示全部app install_mixapps() { local xapk_download_url="https://github.com/wukongdaily/tvhelper/raw/master/apks/mix.xapk" local xapkname=$(basename "$xapk_download_url") local xapk_file="/tmp/$xapkname" wget -O "$xapk_file" "${proxy}$xapk_download_url" local extract_to="/tmp/mix/" mkdir -p "$extract_to" if unzip -o "$xapk_file" -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" # 删除解压目录 rm -f "$xapk_file" # 删除原始XAPK文件 echo -e "${GREEN}临时文件删除完成,行啦,在盒子上查看吧!${NC}" else echo -e "${RED}安装失败${NC}" fi } # 进入KODI助手 kodi_helper() { wget -O kodi.sh "${proxy}https://raw.githubusercontent.com/wukongdaily/tvhelper/master/shells/kodi.sh" && chmod +x kodi.sh && ./kodi.sh ${proxy} } # 安装fire tv版本youtube install_youtube_firetv() { echo -e "${BLUE}Fire TV版本Youtube无需谷歌框架 可用于所有安卓5.0以上电视盒子 ${NC}" install_apk "https://github.com/wukongdaily/tvhelper/raw/master/apks/youtube.apk" "com.amazon.firetv.youtube" } # 进入tvbox安装助手 enter_tvbox_helper() { wget -O /tmp/TVBox.json "${proxy}https://github.com/wukongdaily/tvhelper/raw/master/apks/TVBox.json" wget -O box.sh "${proxy}https://raw.githubusercontent.com/wukongdaily/tvhelper/master/shells/box.sh" && chmod +x box.sh && ./box.sh ${proxy} } # 进入sony电视助手 enter_sonytv() { wget -O sony.sh "${proxy}https://raw.githubusercontent.com/wukongdaily/tvhelper/master/shells/sony.sh" && chmod +x sony.sh && ./sony.sh ${proxy} } install_qrencode() { if ! opkg list-installed | grep -q '^qrencode'; then echo "请稍后..." opkg update >/dev/null 2>&1 opkg install qrencode >/dev/null 2>&1 if [ $? -eq 0 ]; then echo "qrencode安装成功。" else echo "qrencode安装失败。" fi else echo "qrencode已经安装。" fi } # 菜单 menu_options=( "安装ADB" "连接ADB" "断开ADB" "给软路由增加Emotn Store域名" "给软路由添加主机名映射(自定义挟持域名,仅限主路由模式)" "安装Android原生TV必备精选Apps" "一键修改NTP(限原生TV,需重启)" "安装Play商店图标(仅google tv使用)" "自定义批量安装/tmp/upload目录下的所有apk" "替换系统桌面" "进入KODI助手" "进入TVBox安装助手" "进入Sony电视助手" "向TV端输入文字(限英文)" "显示Netflix影片码率" "模拟菜单键" "更新脚本" "赞助|打赏" ) commands=( ["安装ADB"]="install_adb" ["连接ADB"]="connect_adb" ["断开ADB"]="disconnect_adb" ["给软路由增加Emotn Store域名"]="add_emotn_domain" ["给软路由添加主机名映射(自定义挟持域名,仅限主路由模式)"]="add_dhcp_domain" ["安装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" ["自定义批量安装/tmp/upload目录下的所有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() { 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="/tmp/ui.apk" # 检查 com.oversea.aslauncher 是否已安装 if ! adb shell pm list packages | grep -q "$package_name"; then echo -e "${GREEN}EmotnUI 未安装,开始安装...请稍后${NC}" echo -e "${BLUE}正在安装第三方桌面...${NC}" install_apk "https://github.com/wukongdaily/tvhelper/raw/master/apks/ui.apk" "com.oversea.aslauncher" # 再次检查ui是否安装成功,不成功则return if ! adb shell pm list packages | grep -q "$package_name"; then echo -e "${GREEN}EmotnUI 未能安装成功,请调整网络后重试${NC}" return else echo -e "${GREEN}...第三方桌面EmotnUI已安装。${NC}" 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 } update_sh() { echo "正在更新脚本..." # 下载最新的脚本到临时文件 wget -O /tmp/script.sh "${proxy}https://raw.githubusercontent.com/wukongdaily/tvhelper/master/shells/tv.sh" # 替换当前脚本 if [ -f /tmp/script.sh ]; then chmod +x /tmp/script.sh cp /tmp/script.sh /tv.sh echo "脚本更新成功。即将重新启动脚本。" # 使用 exec 来重新启动脚本,替换当前进程 exec /tv.sh ${proxy} else echo "更新失败。" 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" } get_script_mode(){ if [ -n "$proxy" ]; then echo -e " 代理模式" else echo -e "" fi } show_menu() { download_common_shell mkdir -p /tmp/upload clear echo "***********************************************************************" echo -e "* ${YELLOW}盒子助手OP版 当前版本:v${SCRIPT_VERSION}$(get_script_mode)${NC} " echo -e "* ${GREEN}专治安卓原生TV盒子在大陆使用的各种水土不服${NC} " echo -e "* ${RED}请确保电视盒子和OpenWrt路由器处于${NC}${BLUE}同一网段${NC}\n* ${RED}且电视盒子开启了${NC}${BLUE}USB调试模式(adb开关)${NC} " echo "* Developed by @wukongdaily " echo "**********************************************************************" echo echo "* 当前的路由器型号: $(get_router_name)" echo "$(get_status)" echo "$(get_tvbox_model_name)" echo "$(get_tvbox_timezone)" echo 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 echo -e "${GREEN}您已退出悟空的盒子助手,下次运行 ./tv.sh 即可 或者运行 ./tv.sh proxy${NC}" echo break fi handle_choice $choice echo "按任意键继续..." read -n 1 # 等待用户按键 done