You've already forked Tricky-Addon-Update-Target-List
mirror of
https://github.com/KOWX712/Tricky-Addon-Update-Target-List.git
synced 2025-09-06 06:37:09 +00:00
Compare commits
60 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
98ed78cebb | ||
|
|
294a7bf214 | ||
|
|
8f87642319 | ||
|
|
e939f11a72 | ||
|
|
92e654614e | ||
|
|
6b15a09381 | ||
|
|
63c10043dd | ||
|
|
487b2ed434 | ||
|
|
6ed1ec07a8 | ||
|
|
7a07476598 | ||
|
|
5e8d806237 | ||
|
|
ab65ed6520 | ||
|
|
dd786643dc | ||
|
|
c273f1823a | ||
|
|
3d3c47aab4 | ||
|
|
b11fe1dd61 | ||
|
|
6a26150e50 | ||
|
|
1c93287b69 | ||
|
|
d739bc2751 | ||
|
|
defef34bc3 | ||
|
|
e704bda0f7 | ||
|
|
530f006154 | ||
|
|
7a8e5979cd | ||
|
|
48637a0fdb | ||
|
|
ad4cc31c29 | ||
|
|
d5c2fe2cbd | ||
|
|
36ea5f4f99 | ||
|
|
dc84087b33 | ||
|
|
0191078fe5 | ||
|
|
80ffdd3e0d | ||
|
|
9f6faf4e17 | ||
|
|
6419f5e3b9 | ||
|
|
0c659675c5 | ||
|
|
37fce96a11 | ||
|
|
3e36abdbc4 | ||
|
|
9e59a30129 | ||
|
|
b4cd5467ef | ||
|
|
a119c58279 | ||
|
|
1db0259f36 | ||
|
|
4176bd9ce1 | ||
|
|
c879dfa428 | ||
|
|
355c0444c7 | ||
|
|
293f9e1266 | ||
|
|
8e9c7f0db8 | ||
|
|
de5a8b8b87 | ||
|
|
c20e8cde1f | ||
|
|
93e2e8c8ba | ||
|
|
bf726bf863 | ||
|
|
9f859dc488 | ||
|
|
9bed9c0c41 | ||
|
|
211f3d732b | ||
|
|
41ce39be2a | ||
|
|
56ca7ec7a1 | ||
|
|
2784072fb4 | ||
|
|
2a3890e5fb | ||
|
|
59e79b33b7 | ||
|
|
fe76f01439 | ||
|
|
37d78b790e | ||
|
|
30c70d01d6 | ||
|
|
c09f43ab5b |
8
.gitattributes
vendored
8
.gitattributes
vendored
@@ -1,11 +1,5 @@
|
||||
# Declare files that will always have LF line endings on checkout.
|
||||
*.sh text eol=lf
|
||||
*.prop text eol=lf
|
||||
*.md text eol=lf
|
||||
*.xml text eol=lf
|
||||
*.json text eol=lf
|
||||
.extra text eol=lf
|
||||
META-INF/** text eol=lf
|
||||
** text eol=lf
|
||||
|
||||
# Denote all files that are truly binary and should not be modified.
|
||||
module/bin/**/** binary
|
||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -1,2 +1,3 @@
|
||||
__MACOSX
|
||||
.DS_Store
|
||||
applist.json
|
||||
@@ -29,11 +29,10 @@ Configure Tricky Store target.txt with KSU WebUI.
|
||||
| Set verifiedBootHash `optional` | ✅ |
|
||||
| Auto config [security patch](https://github.com/5ec1cff/TrickyStore?tab=readme-ov-file#customize-security-patch-level-121), customizable in WebUI | ✅ |
|
||||
| Provide AOSP Keybox `optional` | ✅ |
|
||||
| Import custom Keybox from device storage | ✅ |
|
||||
| Import custom Keybox from device storage | ✅ |
|
||||
| Add system apps `not recommended` | ✅ |
|
||||
| Valid Keybox `not guaranteed` | ❌ |
|
||||
| Shamiko Whitelist switch. [Why?](https://github.com/rushizgithub/shamiko?tab=readme-ov-file#whitelist) | ❌ |
|
||||
| Periodically add all app to target.txt | ❌ |
|
||||
| Add system apps `GMS added by default` | ❌ |
|
||||
|
||||
## Localization
|
||||
- Read [Translation Guide](https://github.com/KOWX712/Tricky-Addon-Update-Target-List/blob/main/module/webui/locales/A-translate.md)
|
||||
|
||||
27
changelog.md
27
changelog.md
@@ -8,6 +8,33 @@ GitHub release: [Tricky Addon: Update Target List](https://github.com/KOWX712/Tr
|
||||
Telegram channel: [KOW's Little World](https://t.me/kowchannel)
|
||||
|
||||
## Changelog
|
||||
### v3.8
|
||||
- **WebUI:** Optimized UI.
|
||||
- **WebUI:** Added mirror link fallback to fetch content from raw.githubusercontent.com
|
||||
- **WebUI:** Added French translation (#31, @anaelle-dev)
|
||||
- **WebUI:** Added Arabic Translation (#32, @ZG089)
|
||||
- **WebUI:** Added Azerbaijani Translation (#34, @mnasibzade)
|
||||
|
||||
### v3.7
|
||||
- **WebUI:** Optimized UI.
|
||||
- **WebUI:** Added uninstall confirmation dialog.
|
||||
- **WebUI:** Sanitize text content (#23, @totalolage)
|
||||
- **WebUI:** Added selected language memory.
|
||||
- **WebUI:** Fixed Chinese translation (#26 #27, @xiaokuqwq)
|
||||
- **WebUI:** Added Indonesian translation (#28, @ChiseWaguri)
|
||||
- **WebUI:** Added Italian translation (#30, @luigimak)
|
||||
- **MMRL:** Added monet theme suport.
|
||||
- **MMRL:** Fixed fail to display guide when permission is not granted.
|
||||
- No longer add Play Store by default.
|
||||
|
||||
### v3.6
|
||||
- **WebUI:** Option to add system apps.
|
||||
- **WebUI:** Fixed abnormal gap between content and header in MMRL.
|
||||
- Handle some vbmeta related prop, enforce boot hash to lowercase.
|
||||
- No longer auto config security patch by default (old user will remain current setup).
|
||||
- Fixed issue with disable sucompat.
|
||||
- More minor improvements.
|
||||
|
||||
### v3.5
|
||||
- **Script:** Set `system=prop` in auto config.
|
||||
- **WebUI:** Option to fetch secuirty patch date.
|
||||
|
||||
@@ -1,6 +0,0 @@
|
||||
# This file is to pass Minotaur native test 'Partition Check Fail'
|
||||
# Download Key Attestation (chiteroman fork recommended)
|
||||
# Link: https://github.com/chiteroman/KeyAttestation/releases
|
||||
# Get your VerifiedBootHash value from Key Attestation app
|
||||
# Ask here if you don't know how to do: https://t.me/kowchannelchat
|
||||
# Paste verifiedBootHash value on next line and save
|
||||
@@ -2,6 +2,7 @@
|
||||
MODPATH=${0%/*}
|
||||
ORG_PATH="$PATH"
|
||||
SKIPLIST="$MODPATH/tmp/skiplist"
|
||||
XPOSED="$MODPATH/tmp/xposed"
|
||||
|
||||
if [ "$MODPATH" = "/data/adb/modules/.TA_utl/common" ]; then
|
||||
MODDIR="/data/adb/modules/.TA_utl"
|
||||
@@ -26,14 +27,16 @@ download() {
|
||||
}
|
||||
|
||||
get_xposed() {
|
||||
pm list packages -3 | cut -d':' -f2 | grep -vxF -f "$SKIPLIST" | while read -r PACKAGE; do
|
||||
touch "$XPOSED"
|
||||
pm list packages -3 | cut -d':' -f2 | grep -vxF -f "$SKIPLIST" | grep -vxF -f "$XPOSED" | while read -r PACKAGE; do
|
||||
APK_PATH=$(pm path "$PACKAGE" | grep "base.apk" | cut -d':' -f2 | tr -d '\r')
|
||||
if [ -n "$APK_PATH" ]; then
|
||||
if aapt dump xmltree "$APK_PATH" AndroidManifest.xml 2>/dev/null | grep -qE "xposed.category|xposeddescription"; then
|
||||
echo "$PACKAGE"
|
||||
echo "$PACKAGE" >> "$XPOSED"
|
||||
fi
|
||||
fi
|
||||
done
|
||||
cat "$XPOSED"
|
||||
}
|
||||
|
||||
check_update() {
|
||||
@@ -65,12 +68,13 @@ get_update() {
|
||||
}
|
||||
|
||||
install_update() {
|
||||
PATH=/data/adb/ap/bin:/data/adb/ksu/bin:/data/adb/magisk:$PATH
|
||||
if command -v magisk >/dev/null 2>&1; then
|
||||
magisk --install-module "$MODPATH/tmp/module.zip"
|
||||
magisk --install-module "$MODPATH/tmp/module.zip" || exit 1
|
||||
elif command -v apd >/dev/null 2>&1; then
|
||||
apd module install "$MODPATH/tmp/module.zip"
|
||||
apd module install "$MODPATH/tmp/module.zip" || exit 1
|
||||
elif command -v ksud >/dev/null 2>&1; then
|
||||
ksud module install "$MODPATH/tmp/module.zip"
|
||||
ksud module install "$MODPATH/tmp/module.zip" || exit 1
|
||||
else
|
||||
exit 1
|
||||
fi
|
||||
|
||||
@@ -15,9 +15,6 @@ initialize() {
|
||||
cp "$MODPATH/uninstall.sh" "$COMPATH/update/uninstall.sh"
|
||||
fi
|
||||
|
||||
#Set specific path
|
||||
sed -i "s|\"set-path\"|\"/data/adb/modules/$NEW_MODID/\"|" "$MODPATH/webui/scripts/main.js" || abort "! Failed to set path"
|
||||
|
||||
# Set aapt binary
|
||||
cp "$MODPATH/module.prop" "$COMPATH/update/module.prop"
|
||||
mv "$MODPATH/bin/$(getprop ro.product.cpu.abi)/aapt" "$COMPATH/aapt"
|
||||
@@ -32,18 +29,34 @@ find_config() {
|
||||
}
|
||||
|
||||
migrate_config() {
|
||||
# Migrate boot_hash
|
||||
if [ ! -f "/data/adb/boot_hash" ]; then
|
||||
mv "$COMPATH/boot_hash" "/data/adb/boot_hash"
|
||||
else
|
||||
rm -f "$COMPATH/boot_hash"
|
||||
# remove empty file
|
||||
if [ -f "/data/adb/boot_hash" ]; then
|
||||
hash_value=$(grep -v '^#' "/data/adb/boot_hash" | tr -d '[:space:]' | tr '[:upper:]' '[:lower:]')
|
||||
[ -z "$hash_value" ] && rm -f /data/adb/boot_hash || echo "$hash_value" > /data/adb/boot_hash
|
||||
fi
|
||||
|
||||
# Migrate security_patch config*
|
||||
if [ ! -s "/data/adb/security_patch" ]; then
|
||||
echo "#Tricky Addon security patch auto config" > "/data/adb/security_patch"
|
||||
if [ -f "/data/adb/security_patch" ]; then
|
||||
if grep -q "^auto_config=1" "/data/adb/security_patch"; then
|
||||
touch "/data/adb/tricky_store/security_patch_auto_config"
|
||||
fi
|
||||
rm -f "/data/adb/security_patch"
|
||||
fi
|
||||
if ! grep -q "^auto_config=" "/data/adb/security_patch"; then
|
||||
echo "auto_config=1" >> "/data/adb/security_patch"
|
||||
|
||||
# Additional system app
|
||||
if [ ! -f "/data/adb/tricky_store/system_app" ]; then
|
||||
SYSTEM_APP="
|
||||
com.google.android.gms
|
||||
com.google.android.gsf
|
||||
com.android.vending
|
||||
com.oplus.deepthinker
|
||||
com.heytap.speechassist
|
||||
com.coloros.sceneservice"
|
||||
touch "/data/adb/tricky_store/system_app"
|
||||
for app in $SYSTEM_APP; do
|
||||
if pm list packages -s | grep -q "$app"; then
|
||||
echo "$app" >> "/data/adb/tricky_store/system_app"
|
||||
fi
|
||||
done
|
||||
fi
|
||||
}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
id=TA_utl
|
||||
name=Tricky Addon - Update Target List
|
||||
version=v3.5
|
||||
versionCode=350
|
||||
version=v3.8
|
||||
versionCode=380
|
||||
author=KOWX712
|
||||
description=A WebUI to conifgure tricky store target.txt
|
||||
updateJson=https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/update.json
|
||||
|
||||
@@ -22,17 +22,26 @@ add_denylist_to_target() {
|
||||
done
|
||||
}
|
||||
|
||||
resetprop_if_empty() {
|
||||
CURRENT=$(getprop "$1")
|
||||
[ -z "$CURRENT" ] && resetprop -n "$1" "$2"
|
||||
}
|
||||
|
||||
# Spoof security patch
|
||||
if grep -q "^auto_config=1" "/data/adb/security_patch"; then
|
||||
if [ -f "/data/adb/tricky_store/security_patch_auto_config" ]; then
|
||||
sh "$MODPATH/common/get_extra.sh" --security-patch
|
||||
fi
|
||||
|
||||
# Reset verified Boot Hash
|
||||
hash_value=$(grep -v '^#' "/data/adb/boot_hash" | tr -d '[:space:]')
|
||||
if [ -n "$hash_value" ]; then
|
||||
resetprop -n ro.boot.vbmeta.digest "$hash_value"
|
||||
# Reset vbmeta related prop
|
||||
if [ -f "/data/adb/boot_hash" ]; then
|
||||
hash_value=$(grep -v '^#' "/data/adb/boot_hash" | tr -d '[:space:]' | tr '[:upper:]' '[:lower:]')
|
||||
[ -z "$hash_value" ] && rm -f /data/adb/boot_hash || resetprop -n ro.boot.vbmeta.digest "$hash_value"
|
||||
fi
|
||||
|
||||
resetprop_if_empty "ro.boot.vbmeta.device_state" "locked"
|
||||
resetprop_if_empty "ro.boot.vbmeta.invalidate_on_error" "yes"
|
||||
resetprop_if_empty "ro.boot.vbmeta.avb_version" "1.0"
|
||||
resetprop_if_empty "ro.boot.vbmeta.hash_alg" "sha256"
|
||||
resetprop_if_empty "ro.boot.vbmeta.size" "10496"
|
||||
|
||||
# Disable TSupport-A auto update target to prevent overwrite
|
||||
if [ -d "$TSPA" ]; then
|
||||
@@ -46,10 +55,12 @@ if [ -f "$MODPATH/action.sh" ]; then
|
||||
# Hide module from Magisk manager
|
||||
if [ "$MODPATH" != "$HIDE_DIR" ]; then
|
||||
rm -rf "$HIDE_DIR"
|
||||
mv "$MODPATH" "$HIDE_DIR"
|
||||
mkdir -p "$HIDE_DIR"
|
||||
busybox chcon --reference="$MODPATH" "$HIDE_DIR"
|
||||
cp -af "$MODPATH/." "$HIDE_DIR/"
|
||||
fi
|
||||
MODPATH="$HIDE_DIR"
|
||||
|
||||
|
||||
# Add target from denylist
|
||||
# To trigger this, choose "Select from DenyList" in WebUI once
|
||||
[ -f "/data/adb/tricky_store/target_from_denylist" ] && add_denylist_to_target
|
||||
@@ -61,16 +72,17 @@ fi
|
||||
rm -f "$MODPATH/module.prop"
|
||||
|
||||
# Symlink tricky store
|
||||
if [ -f "$MODPATH/action.sh" ] && [ ! -f "$TS/action.sh" ] && [ ! -L "$TS/action.sh" ]; then
|
||||
if [ -f "$MODPATH/action.sh" ] && [ ! -e "$TS/action.sh" ]; then
|
||||
ln -s "$MODPATH/action.sh" "$TS/action.sh"
|
||||
fi
|
||||
if [ ! -d "$TS/webroot" ] && [ ! -L "$TS/webroot" ]; then
|
||||
if [ ! -e "$TS/webroot" ]; then
|
||||
ln -s "$MODPATH/webui" "$TS/webroot"
|
||||
fi
|
||||
|
||||
# Optimization
|
||||
OUTPUT_APP="$MODPATH/common/tmp/applist"
|
||||
OUTPUT_APP="$MODPATH/webui/applist.json"
|
||||
OUTPUT_SKIP="$MODPATH/common/tmp/skiplist"
|
||||
OUTPUT_XPOSED="$MODPATH/common/tmp/xposed"
|
||||
|
||||
until [ "$(getprop sys.boot_completed)" = "1" ]; do
|
||||
sleep 1
|
||||
@@ -80,32 +92,40 @@ done
|
||||
mkdir -p "$MODPATH/common/tmp"
|
||||
|
||||
# Additional system apps
|
||||
SYSTEM_APP="com.google.android.gms|com.google.android.gsf|com.android.vending"
|
||||
if [ -f "/data/adb/tricky_store/system_app" ]; then
|
||||
SYSTEM_APP=$(cat "/data/adb/tricky_store/system_app" | tr '\n' '|' | sed 's/|*$//')
|
||||
else
|
||||
SYSTEM_APP=""
|
||||
fi
|
||||
|
||||
# Initialize cache files to save app list and skip list
|
||||
echo "# This file is generated from service.sh to speed up load time" > "$OUTPUT_APP"
|
||||
echo "[" > "$OUTPUT_APP"
|
||||
echo "# This file is generated from service.sh to speed up load time" > "$OUTPUT_SKIP"
|
||||
|
||||
# Get list of third party apps and specific system apps, then cache app name
|
||||
# Check Xposed module
|
||||
{
|
||||
pm list packages -3 2>/dev/null
|
||||
pm list package -s | grep -E "$SYSTEM_APP" 2>/dev/null || true
|
||||
pm list packages -s | grep -E "$SYSTEM_APP" 2>/dev/null || true
|
||||
} | awk -F: '{print $2}' | while read -r PACKAGE; do
|
||||
# Get APK path for the package
|
||||
APK_PATH=$(pm path "$PACKAGE" 2>/dev/null | grep "base.apk" | awk -F: '{print $2}' | tr -d '\r')
|
||||
[ -z "$APK_PATH" ] && APK_PATH=$(pm path "$PACKAGE" 2>/dev/null | grep ".apk" | awk -F: '{print $2}' | tr -d '\r')
|
||||
APK_PATH=$(pm path "$PACKAGE" 2>/dev/null | grep "base.apk" | awk -F: '{print $2}')
|
||||
[ -z "$APK_PATH" ] && APK_PATH=$(pm path "$PACKAGE" 2>/dev/null | grep ".apk" | awk -F: '{print $2}')
|
||||
|
||||
if [ -n "$APK_PATH" ]; then
|
||||
# Extract app name and save package info
|
||||
APP_NAME=$(aapt dump badging "$APK_PATH" 2>/dev/null | grep "application-label:" | sed "s/application-label://g; s/'//g")
|
||||
echo "app-name: $APP_NAME, package-name: $PACKAGE" >> "$OUTPUT_APP"
|
||||
else
|
||||
echo "app-name: Unknown App package-name: $PACKAGE" >> "$OUTPUT_APP"
|
||||
fi
|
||||
APP_NAME=$(aapt dump badging "$APK_PATH" 2>/dev/null | grep "application-label:" | sed "s/application-label://g; s/'//g")
|
||||
[ -z "$APP_NAME" ] && APP_NAME="$PACKAGE"
|
||||
echo " {\"app_name\": \"$APP_NAME\", \"package_name\": \"$PACKAGE\"}," >> "$OUTPUT_APP"
|
||||
|
||||
# Check if app is Xposed module and add to skip list if not
|
||||
if ! aapt dump xmltree "$APK_PATH" AndroidManifest.xml 2>/dev/null | grep -qE "xposed.category|xposeddescription"; then
|
||||
touch "$OUTPUT_XPOSED"
|
||||
if aapt dump xmltree "$APK_PATH" AndroidManifest.xml 2>/dev/null | grep -qE "xposed.category|xposeddescription"; then
|
||||
echo "$PACKAGE" >> "$OUTPUT_XPOSED"
|
||||
else
|
||||
echo "$PACKAGE" >> "$OUTPUT_SKIP"
|
||||
fi
|
||||
done
|
||||
|
||||
sed -i '$ s/,$//' "$OUTPUT_APP"
|
||||
echo "]" >> "$OUTPUT_APP"
|
||||
|
||||
[ -f "$MODPATH/action.sh" ] && rm -rf "/data/adb/modules/TA_utl"
|
||||
|
||||
@@ -10,8 +10,9 @@ fi
|
||||
# Remove residue and restore aosp keybox.
|
||||
rm -rf "/data/adb/modules/.TA_utl"
|
||||
rm -f "/data/adb/boot_hash"
|
||||
rm -f "/data/adb/security_patch"
|
||||
rm -f "/data/adb/tricky_store/security_patch_auto_config"
|
||||
rm -f "/data/adb/tricky_store/target_from_denylist"
|
||||
rm -f "/data/adb/tricky_store/system_app"
|
||||
if [ -d "$TS" ]; then
|
||||
[ -L "$TS/webroot" ] && rm -f "$TS/webroot"
|
||||
[ -L "$TS/action.sh" ] && rm -f "$TS/action.sh"
|
||||
|
||||
@@ -4,15 +4,16 @@
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title data-i18n="title">TrickyAddon</title>
|
||||
<link rel="stylesheet" type="text/css" href="/mmrl/insets.css" />
|
||||
<title>TrickyAddon</title>
|
||||
<link rel="stylesheet" href="styles/global.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/about.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/applist.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/boot_hash.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/file_selector.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/header.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/search_menu.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/security_patch.css" type="text/css">
|
||||
<link rel="stylesheet" href="styles/system_app.css" type="text/css">
|
||||
<script type="module" crossorigin src="scripts/main.js"></script>
|
||||
<script type="module" crossorigin src="scripts/about.js"></script>
|
||||
<script type="module" crossorigin src="scripts/help.js"></script>
|
||||
@@ -23,7 +24,6 @@
|
||||
|
||||
<body>
|
||||
<!-- Header -->
|
||||
<div class="header-block"></div>
|
||||
<div class="header">
|
||||
<div id="title" data-i18n="header.title"></div><span id="module-version"></span>
|
||||
<button id="help-button" class="help-button">
|
||||
@@ -77,6 +77,7 @@
|
||||
<li class="ripple-element" id="deselect-all" data-i18n="menu.deselect_all"></li>
|
||||
<li class="ripple-element" id="select-denylist" data-i18n="menu.select_denylist"></li>
|
||||
<li class="ripple-element" id="deselect-unnecessary" data-i18n="menu.deselect_unnecessary"></li>
|
||||
<li class="ripple-element" id="add-system-app" data-i18n="menu.add_system_app"></li>
|
||||
<li class="ripple-element" id="aospkb" data-i18n="menu.set_aosp_keybox"></li>
|
||||
<li class="ripple-element" id="validkb" data-i18n="menu.set_valid_keybox"></li>
|
||||
<li class="ripple-element" id="customkb" data-i18n="menu.set_custom_keybox"></li>
|
||||
@@ -102,29 +103,29 @@
|
||||
<div class="content" data-package="">
|
||||
<div class="mode">
|
||||
<label class="mode-switch" id="normal">
|
||||
<input type="radio" class="mode-input" id="normal-mode">
|
||||
<i class="mode-icon">
|
||||
<div class="status-indicator ripple-element" id="normal-indicator">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="30px" viewBox="0 -960 960 960" width="30px" fill="#ffffff"><path d="M480-480Zm0 280q-116 0-198-82t-82-198q0-116 82-198t198-82q116 0 198 82t82 198q0 116-82 198t-198 82Zm0-80q83 0 141.5-58.5T680-480q0-83-58.5-141.5T480-680q-83 0-141.5 58.5T280-480q0 83 58.5 141.5T480-280Z"/></svg>
|
||||
</div>
|
||||
</i>
|
||||
</label>
|
||||
<label class="mode-switch" id="generate">
|
||||
<input type="radio" class="mode-input" id="generate-mode">
|
||||
<i class="mode-icon">
|
||||
<div class="status-indicator ripple-element" id="generate-indicator">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="18px" viewBox="0 -960 960 960" width="18px" fill="#ffffff"><path d="M480-120q-33 0-56.5-23.5T400-200q0-33 23.5-56.5T480-280q33 0 56.5 23.5T560-200q0 33-23.5 56.5T480-120Zm-80-240v-480h160v480H400Z"/></svg>
|
||||
</div>
|
||||
</i>
|
||||
</label>
|
||||
<label class="mode-switch" id="hack">
|
||||
<input type="radio" class="mode-input" id="hack-mode">
|
||||
<i class="mode-icon">
|
||||
<div class="status-indicator ripple-element" id="hack-indicator">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="19px" viewBox="0 -960 960 960" width="19px" fill="#ffffff"><path d="M424-320q0-81 14.5-116.5T500-514q41-36 62.5-62.5T584-637q0-41-27.5-68T480-732q-51 0-77.5 31T365-638l-103-44q21-64 77-111t141-47q105 0 161.5 58.5T698-641q0 50-21.5 85.5T609-475q-49 47-59.5 71.5T539-320H424Zm56 240q-33 0-56.5-23.5T400-160q0-33 23.5-56.5T480-240q33 0 56.5 23.5T560-160q0 33-23.5 56.5T480-80Z"/></svg>
|
||||
</div>
|
||||
</i>
|
||||
</label>
|
||||
<input type="radio" class="mode-input" id="normal-mode">
|
||||
<i class="mode-icon">
|
||||
<div class="status-indicator ripple-element" id="normal-indicator">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="30px" viewBox="0 -960 960 960" width="30px" fill="#ffffff"><path d="M480-480Zm0 280q-116 0-198-82t-82-198q0-116 82-198t198-82q116 0 198 82t82 198q0 116-82 198t-198 82Zm0-80q83 0 141.5-58.5T680-480q0-83-58.5-141.5T480-680q-83 0-141.5 58.5T280-480q0 83 58.5 141.5T480-280Z"/></svg>
|
||||
</div>
|
||||
</i>
|
||||
</label>
|
||||
<label class="mode-switch" id="generate">
|
||||
<input type="radio" class="mode-input" id="generate-mode">
|
||||
<i class="mode-icon">
|
||||
<div class="status-indicator ripple-element" id="generate-indicator">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="18px" viewBox="0 -960 960 960" width="18px" fill="#ffffff"><path d="M480-120q-33 0-56.5-23.5T400-200q0-33 23.5-56.5T480-280q33 0 56.5 23.5T560-200q0 33-23.5 56.5T480-120Zm-80-240v-480h160v480H400Z"/></svg>
|
||||
</div>
|
||||
</i>
|
||||
</label>
|
||||
<label class="mode-switch" id="hack">
|
||||
<input type="radio" class="mode-input" id="hack-mode">
|
||||
<i class="mode-icon">
|
||||
<div class="status-indicator ripple-element" id="hack-indicator">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="19px" viewBox="0 -960 960 960" width="19px" fill="#ffffff"><path d="M424-320q0-81 14.5-116.5T500-514q41-36 62.5-62.5T584-637q0-41-27.5-68T480-732q-51 0-77.5 31T365-638l-103-44q21-64 77-111t141-47q105 0 161.5 58.5T698-641q0 50-21.5 85.5T609-475q-49 47-59.5 71.5T539-320H424Zm56 240q-33 0-56.5-23.5T400-160q0-33 23.5-56.5T480-240q33 0 56.5 23.5T560-160q0 33-23.5 56.5T480-80Z"/></svg>
|
||||
</div>
|
||||
</i>
|
||||
</label>
|
||||
</div>
|
||||
<p class="name"></p>
|
||||
<div class="checkbox-wrapper">
|
||||
@@ -141,112 +142,85 @@
|
||||
</template>
|
||||
|
||||
<!-- Help Overlay -->
|
||||
<div id="help-overlay" class="help-overlay">
|
||||
<div class="help-menu">
|
||||
<button id="close-help" class="close-help">✕</button>
|
||||
<div id="help-overlay" class="help-overlay overlay">
|
||||
<div class="help-menu overlay-content">
|
||||
<button id="close-help" class="close-btn">✕</button>
|
||||
<div class="help-content">
|
||||
<p data-i18n="help.help_instructions"></p>
|
||||
<ul id="helpList">
|
||||
<li id="save_and_update_button">
|
||||
<strong data-i18n="help.save_and_update"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.save_and_update_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="refresh">
|
||||
<strong data-i18n="help.refresh"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.refresh_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="select_deselect">
|
||||
<strong data-i18n="help.select_deselect"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.select_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="select_denylist">
|
||||
<strong data-i18n="help.select_denylist"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.select_denylist_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="deselect_unnecessary">
|
||||
<strong data-i18n="help.deselect_unnecessary"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.deselect_unnecessary_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="set_keybox">
|
||||
<strong data-i18n="help.set_keybox"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.set_keybox_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="set_custom_keybox">
|
||||
<strong data-i18n="help.set_custom_keybox"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.set_custom_keybox_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="set_security_patch">
|
||||
<strong data-i18n="help.set_security_patch"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.set_security_patch_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
<li id="set_verified_boot_hash">
|
||||
<strong data-i18n="help.set_verified_boot_hash"></strong>
|
||||
<ul>
|
||||
<li data-i18n="help.set_verified_boot_hash_description"></li>
|
||||
</ul>
|
||||
</li>
|
||||
<br>
|
||||
</ul>
|
||||
<div class="help-content-header" data-i18n="help.help_instructions"></div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.save_and_update"></strong>
|
||||
<p data-i18n="help.save_and_update_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.refresh"></strong>
|
||||
<p data-i18n="help.refresh_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.select_deselect"></strong>
|
||||
<p data-i18n="help.select_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.select_denylist"></strong>
|
||||
<p data-i18n="help.select_denylist_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.deselect_unnecessary"></strong>
|
||||
<p data-i18n="help.deselect_unnecessary_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.add_system_app"></strong>
|
||||
<p data-i18n="help.add_system_app_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.set_keybox"></strong>
|
||||
<p data-i18n="help.set_keybox_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.set_custom_keybox"></strong>
|
||||
<p data-i18n="help.set_custom_keybox_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.set_security_patch"></strong>
|
||||
<p data-i18n="help.set_security_patch_description"></p>
|
||||
</div>
|
||||
<div class="instruction">
|
||||
<strong data-i18n="help.set_verified_boot_hash"></strong>
|
||||
<p data-i18n="help.set_verified_boot_hash_description"></p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- BootHash Input Overlay -->
|
||||
<div id="boot-hash-overlay" class="boot-hash-overlay"></div>
|
||||
<div id="boot-hash-card" class="boot-hash-card">
|
||||
<div class="boot-hash-title" data-i18n="boot_hash.title"></div>
|
||||
<textarea id="boot-hash-input" class="boot-hash-input" placeholder="Paste your verified Boot Hash here" data-i18n="boot_hash.input_placeholder" oninput="window.trimInput(this)"></textarea>
|
||||
<button id="boot-hash-save-button" class="boot-hash-save-button ripple-element" data-i18n="boot_hash.save_button"></button>
|
||||
<div id="boot-hash-overlay" class="boot-hash-overlay overlay">
|
||||
<div id="boot-hash-card" class="boot-hash-card overlay-content">
|
||||
<div class="boot-hash-title" data-i18n="boot_hash.title"></div>
|
||||
<textarea id="boot-hash-input" class="boot-hash-input" placeholder="Paste your verified Boot Hash here" data-i18n="boot_hash.input_placeholder" oninput="window.trimInput(this)"></textarea>
|
||||
<button id="boot-hash-save-button" class="boot-hash-save-button ripple-element" data-i18n="boot_hash.save_button"></button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- About Overlay -->
|
||||
<div id="about-overlay" class="about-overlay">
|
||||
<div id="about-menu" class="about-menu">
|
||||
<button id="close-about" class="close-about">✕</button>
|
||||
<div class="about-content">
|
||||
<div id="about-overlay" class="about-overlay overlay">
|
||||
<div class="about-menu overlay-content">
|
||||
<button id="close-about" class="close-btn">✕</button>
|
||||
<div class="about-title">
|
||||
<p id="module_name_line1" data-i18n="about.module_name_line1"></p>
|
||||
<p id="module_name_line2" data-i18n="about.module_name_line2"></p>
|
||||
<p><span id="authored" data-i18n="about.by"></span> KOWX712</p>
|
||||
<br>
|
||||
<p id="disclaimer" data-i18n="about.disclaimer"></p>
|
||||
<br>
|
||||
<p>
|
||||
<div class="link">
|
||||
<i class="link-icon ripple-element" id="telegram" aria-hidden="true">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="20px" viewBox="0 0 496 512"><!--!Font Awesome Free 6.7.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M248 8C111 8 0 119 0 256S111 504 248 504 496 393 496 256 385 8 248 8zM363 176.7c-3.7 39.2-19.9 134.4-28.1 178.3-3.5 18.6-10.3 24.8-16.9 25.4-14.4 1.3-25.3-9.5-39.3-18.7-21.8-14.3-34.2-23.2-55.3-37.2-24.5-16.1-8.6-25 5.3-39.5 3.7-3.8 67.1-61.5 68.3-66.7 .2-.7 .3-3.1-1.2-4.4s-3.6-.8-5.1-.5q-3.3 .7-104.6 69.1-14.8 10.2-26.9 9.9c-8.9-.2-25.9-5-38.6-9.1-15.5-5-27.9-7.7-26.8-16.3q.8-6.7 18.5-13.7 108.4-47.2 144.6-62.3c68.9-28.6 83.2-33.6 92.5-33.8 2.1 0 6.6 .5 9.6 2.9a10.5 10.5 0 0 1 3.5 6.7A43.8 43.8 0 0 1 363 176.7z"/></svg>
|
||||
<span id="link-text" data-i18n="about.telegram_channel"></span>
|
||||
</i>
|
||||
<i class="link-icon ripple-element" id="github" aria-hidden="true">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="20px" viewBox="0 0 496 512"><!--!Font Awesome Free 6.7.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M165.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3 .3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6zm-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5 .3-6.2 2.3zm44.2-1.7c-2.9 .7-4.9 2.6-4.6 4.9 .3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9zM244.8 8C106.1 8 0 113.3 0 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C428.2 457.8 496 362.9 496 252 496 113.3 383.5 8 244.8 8zM97.2 352.9c-1.3 1-1 3.3 .7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1zm-10.8-8.1c-.7 1.3 .3 2.9 2.3 3.9 1.6 1 3.6 .7 4.3-.7 .7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3 .7zm32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3 .7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1zm-11.4-14.7c-1.6 1-1.6 3.6 0 5.9 1.6 2.3 4.3 3.3 5.6 2.3 1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2z"/></svg>
|
||||
<span id="link-text" data-i18n="about.github"></span>
|
||||
</i>
|
||||
</div>
|
||||
</p>
|
||||
<br>
|
||||
</div>
|
||||
<div id="disclaimer" data-i18n="about.disclaimer"></div>
|
||||
<div class="link">
|
||||
<i class="link-icon ripple-element" id="telegram" aria-hidden="true">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="20px" viewBox="0 0 496 512"><!--!Font Awesome Free 6.7.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M248 8C111 8 0 119 0 256S111 504 248 504 496 393 496 256 385 8 248 8zM363 176.7c-3.7 39.2-19.9 134.4-28.1 178.3-3.5 18.6-10.3 24.8-16.9 25.4-14.4 1.3-25.3-9.5-39.3-18.7-21.8-14.3-34.2-23.2-55.3-37.2-24.5-16.1-8.6-25 5.3-39.5 3.7-3.8 67.1-61.5 68.3-66.7 .2-.7 .3-3.1-1.2-4.4s-3.6-.8-5.1-.5q-3.3 .7-104.6 69.1-14.8 10.2-26.9 9.9c-8.9-.2-25.9-5-38.6-9.1-15.5-5-27.9-7.7-26.8-16.3q.8-6.7 18.5-13.7 108.4-47.2 144.6-62.3c68.9-28.6 83.2-33.6 92.5-33.8 2.1 0 6.6 .5 9.6 2.9a10.5 10.5 0 0 1 3.5 6.7A43.8 43.8 0 0 1 363 176.7z"/></svg>
|
||||
<span id="link-text" data-i18n="about.telegram_channel"></span>
|
||||
</i>
|
||||
<i class="link-icon ripple-element" id="github" aria-hidden="true">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="20px" viewBox="0 0 496 512"><!--!Font Awesome Free 6.7.2 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M165.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3 .3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6zm-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5 .3-6.2 2.3zm44.2-1.7c-2.9 .7-4.9 2.6-4.6 4.9 .3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9zM244.8 8C106.1 8 0 113.3 0 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C428.2 457.8 496 362.9 496 252 496 113.3 383.5 8 244.8 8zM97.2 352.9c-1.3 1-1 3.3 .7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1zm-10.8-8.1c-.7 1.3 .3 2.9 2.3 3.9 1.6 1 3.6 .7 4.3-.7 .7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3 .7zm32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3 .7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1zm-11.4-14.7c-1.6 1-1.6 3.6 0 5.9 1.6 2.3 4.3 3.3 5.6 2.3 1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2z"/></svg>
|
||||
<span id="link-text" data-i18n="about.github"></span>
|
||||
</i>
|
||||
</div>
|
||||
<div class="acknowledgment">
|
||||
<p id="acknowledgment" data-i18n="about.acknowledgment"></p>
|
||||
<p>j-hc/zygisk-detach: WebUI template</p>
|
||||
<p>markedjs/marked: Markdown Support</p>
|
||||
@@ -255,9 +229,9 @@
|
||||
</div>
|
||||
|
||||
<!-- Update Overlay -->
|
||||
<div class="update-overlay">
|
||||
<div class="update-menu">
|
||||
<button class="close-update">✕</button>
|
||||
<div class="update-overlay overlay">
|
||||
<div class="update-menu overlay-content">
|
||||
<button id="close-update" class="close-btn">✕</button>
|
||||
<div class="update-content">
|
||||
<h1 data-i18n="update.changelog"></h1>
|
||||
<div class="changelog"></div>
|
||||
@@ -270,64 +244,106 @@
|
||||
</div>
|
||||
|
||||
<!-- MMRL Permission Request Overlay -->
|
||||
<div id="permission-popup" class="permission-popup hidden">
|
||||
<div id="permission-popup" class="permission-popup overlay">
|
||||
<div class="permission-content">
|
||||
<h2 id="permission-title">Please allow JavaScript API in MMRL settings</h2>
|
||||
<div class="permission-steps">
|
||||
<p>1. Settings</p>
|
||||
<p>2. Security</p>
|
||||
<p>3. Allow JavaScript API</p>
|
||||
<p>4. Tricky Store</p>
|
||||
<p>5. Enable Allow Advanced KernelSU API</p>
|
||||
<ol>
|
||||
<li>Settings</li>
|
||||
<li>Security</li>
|
||||
<li>Allow JavaScript API</li>
|
||||
<li>Tricky Store</li>
|
||||
<li>Enable "Allow Advanced KernelSU API"</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- File Selector Overlay -->
|
||||
<div class="file-selector-overlay overlay">
|
||||
<div class="file-selector overlay-content">
|
||||
<div class="file-selector-header">
|
||||
<button class="back-button">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="22px" viewBox="40 -1050 900 960" width="22px"><path d="M400-93.85 13.85-480 400-866.15l56.77 56.77L127.38-480l329.39 329.38L400-93.85Z"/></svg>
|
||||
</button>
|
||||
<div class="current-path">/storage/emulated/0/Download</div>
|
||||
<button class="close-selector">✕</button>
|
||||
</div>
|
||||
<div class="file-list"></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Security Patch Overlay -->
|
||||
<div id="security-patch-overlay" class="security-patch-overlay"></div>
|
||||
<div id="security-patch-card" class="security-patch-card">
|
||||
<div class="security-patch-header" data-i18n="security_patch.title"></div>
|
||||
<div class="security-patch-content">
|
||||
<div id="normal-mode-inputs" class="normal-mode-inputs">
|
||||
<div class="input-group">
|
||||
<label id="security_patch-all">All</label>
|
||||
<input type="text" id="all-patch" placeholder="20250101" maxlength="8" autocapitalize="none">
|
||||
<div id="security-patch-overlay" class="security-patch-overlay overlay">
|
||||
<div id="security-patch-card" class="security-patch-card overlay-content">
|
||||
<div class="security-patch-header" data-i18n="security_patch.title"></div>
|
||||
<div class="security-patch-content">
|
||||
<div id="normal-mode-inputs" class="normal-mode-inputs">
|
||||
<div class="input-group">
|
||||
<label id="security_patch-all">All</label>
|
||||
<input type="text" id="all-patch" placeholder="20250101" maxlength="8" autocapitalize="none">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="advanced-mode-inputs" class="advanced-mode-inputs hidden">
|
||||
<div class="input-group">
|
||||
<label id="security_patch-system">System</label>
|
||||
<input type="text" id="system-patch" placeholder="202501" maxlength="6" autocapitalize="none">
|
||||
</div>
|
||||
<div class="input-group">
|
||||
<label id="security_patch-boot">Boot</label>
|
||||
<input type="text" id="boot-patch" placeholder="2025-01-01" autocapitalize="none" oninput="formatDate(this, 'boot')" maxlength="10">
|
||||
</div>
|
||||
<div class="input-group">
|
||||
<label id="security_patch-vendor">Vendor</label>
|
||||
<input type="text" id="vendor-patch" placeholder="2025-01-01" autocapitalize="none" oninput="formatDate(this, 'vendor')" maxlength="10">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="advanced-toggle">
|
||||
<input type="checkbox" class="checkbox" id="advanced-mode" />
|
||||
<label for="advanced-mode" class="custom-checkbox">
|
||||
<span class="tick-symbol">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 -3 26 26" width="16px" height="16px" fill="#fff">
|
||||
<path d="M 22.566406 4.730469 L 20.773438 3.511719 C 20.277344 3.175781 19.597656 3.304688 19.265625 3.796875 L 10.476563 16.757813 L 6.4375 12.71875 C 6.015625 12.296875 5.328125 12.296875 4.90625 12.71875 L 3.371094 14.253906 C 2.949219 14.675781 2.949219 15.363281 3.371094 15.789063 L 9.582031 22 C 9.929688 22.347656 10.476563 22.613281 10.96875 22.613281 C 11.460938 22.613281 11.957031 22.304688 12.277344 21.839844 L 22.855469 6.234375 C 23.191406 5.742188 23.0625 5.066406 22.566406 4.730469 Z"/>
|
||||
</svg>
|
||||
</span>
|
||||
</label>
|
||||
<label for="advanced-mode" data-i18n="security_patch.advanced_mode"></label>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<button id="get-patch" class="get-button ripple-element" data-i18n="security_patch.get_date"></button>
|
||||
<div class="button-container">
|
||||
<button id="auto-config" class="auto-button ripple-element" data-i18n="security_patch.auto"></button>
|
||||
<button id="save-patch" class="save-button ripple-element" data-i18n="security_patch.save"></button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="advanced-mode-inputs" class="advanced-mode-inputs hidden">
|
||||
<div class="input-group">
|
||||
<label id="security_patch-system">System</label>
|
||||
<input type="text" id="system-patch" placeholder="202501" maxlength="6" autocapitalize="none">
|
||||
</div>
|
||||
<div class="input-group">
|
||||
<label id="security_patch-boot">Boot</label>
|
||||
<input type="text" id="boot-patch" placeholder="2025-01-01" autocapitalize="none" oninput="formatDate(this, 'boot')" maxlength="10">
|
||||
</div>
|
||||
<div class="input-group">
|
||||
<label id="security_patch-vendor">Vendor</label>
|
||||
<input type="text" id="vendor-patch" placeholder="2025-01-01" autocapitalize="none" oninput="formatDate(this, 'vendor')" maxlength="10">
|
||||
</div>
|
||||
<!-- Add System App Overlay -->
|
||||
<div id="add-system-app-overlay" class="add-system-app-overlay overlay">
|
||||
<div id="add-system-app-card" class="add-system-app-card overlay-content">
|
||||
<div class="add-system-app-title" data-i18n="add_system_app.title"></div>
|
||||
<div class="add-system-app-content">
|
||||
<input type="text" id="system-app-input" placeholder="com.example.app" autocapitalize="none">
|
||||
<button id="add-system-app-button" class="add-system-app-button ripple-element" data-i18n="add_system_app.add"></button>
|
||||
<h3 class="current-system-app-list" data-i18n="add_system_app.current_list"></h3>
|
||||
<div class="current-system-app-list-content"></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="advanced-toggle">
|
||||
<input type="checkbox" class="checkbox" id="advanced-mode" />
|
||||
<label for="advanced-mode" class="custom-checkbox">
|
||||
<span class="tick-symbol">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 -3 26 26" width="16px" height="16px" fill="#fff">
|
||||
<path d="M 22.566406 4.730469 L 20.773438 3.511719 C 20.277344 3.175781 19.597656 3.304688 19.265625 3.796875 L 10.476563 16.757813 L 6.4375 12.71875 C 6.015625 12.296875 5.328125 12.296875 4.90625 12.71875 L 3.371094 14.253906 C 2.949219 14.675781 2.949219 15.363281 3.371094 15.789063 L 9.582031 22 C 9.929688 22.347656 10.476563 22.613281 10.96875 22.613281 C 11.460938 22.613281 11.957031 22.304688 12.277344 21.839844 L 22.855469 6.234375 C 23.191406 5.742188 23.0625 5.066406 22.566406 4.730469 Z"/>
|
||||
</svg>
|
||||
</span>
|
||||
</label>
|
||||
<label for="advanced-mode" data-i18n="security_patch.advanced_mode"></label>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<button id="get-patch" class="get-button ripple-element" data-i18n="security_patch.get_date"></button>
|
||||
<div class="button-container">
|
||||
<button id="auto-config" class="auto-button ripple-element" data-i18n="security_patch.auto"></button>
|
||||
<button id="save-patch" class="save-button ripple-element" data-i18n="security_patch.save"></button>
|
||||
</div>
|
||||
<!-- Uninstall Confirmation Overlay -->
|
||||
<div class="uninstall-confirmation-overlay overlay" id="uninstall-confirmation-overlay">
|
||||
<div class="uninstall-confirmation overlay-content">
|
||||
<div class="uninstall-confirmation-title" data-i18n="confirmation.uninstall_title"></div>
|
||||
<p data-i18n="confirmation.uninstall_message"></p>
|
||||
<div class="uninstall-confirmation-button-container">
|
||||
<button class="uninstall-confirmation-button ripple-element" id="cancel-uninstall" data-i18n="confirmation.uninstall_cancel"></button>
|
||||
<button class="uninstall-confirmation-button ripple-element" id="confirm-uninstall" data-i18n="confirmation.uninstall_confirm"></button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -335,7 +351,7 @@
|
||||
<!-- Footer -->
|
||||
<div class="footer">
|
||||
<div class="uninstall-container ripple-element hidden-uninstall">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="24px" viewBox="0 -960 960 960" width="24px" fill="#FF3636"><path d="M280-120q-33 0-56.5-23.5T200-200v-520h-40v-80h200v-40h240v40h200v80h-40v520q0 33-23.5 56.5T680-120H280Zm400-600H280v520h400v-520ZM360-280h80v-360h-80v360Zm160 0h80v-360h-80v360ZM280-720v520-520Z" /></svg>
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="24px" viewBox="0 -960 960 960" width="24px"><path d="M280-120q-33 0-56.5-23.5T200-200v-520h-40v-80h200v-40h240v40h200v80h-40v520q0 33-23.5 56.5T680-120H280Zm400-600H280v520h400v-520ZM360-280h80v-360h-80v360Zm160 0h80v-360h-80v360ZM280-720v520-520Z" /></svg>
|
||||
<span data-i18n="functional_button.uninstall_webui"></span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "English",
|
||||
"system_default": "System Default",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Available in Magisk only, select apps that are in the DenyList. Recommended.",
|
||||
"deselect_unnecessary": "Deselect Unnecessary",
|
||||
"deselect_unnecessary_description": "Unnecessary category: Xposed module, root manager, root-related apps, and general apps that never check bootloader status. This option requires Internet connection.",
|
||||
"add_system_app": "Add System App",
|
||||
"add_system_app_description": "Add specific system app to app list.",
|
||||
"set_keybox": "Set AOSP & Valid Keybox",
|
||||
"set_keybox_description": "Replace tricky store keybox.xml. AOSP keybox will be replaced if there's no more valid keybox. Valid keybox option requires Internet connection.",
|
||||
"set_custom_keybox": "Set Custom Keybox",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Deselect All",
|
||||
"select_denylist": "Select From DenyList",
|
||||
"deselect_unnecessary": "Deselect Unnecessary",
|
||||
"add_system_app": "Add System App",
|
||||
"set_aosp_keybox": "Set AOSP Keybox",
|
||||
"set_valid_keybox": "Set Valid Keybox",
|
||||
"set_custom_keybox": "Set Custom Keybox",
|
||||
@@ -86,12 +90,14 @@
|
||||
"download_fail": "Fail to download update",
|
||||
"installing": "Installing update...",
|
||||
"installed": "Installed successfully, reboot now.",
|
||||
"install_fail": "Fail to install, please update manual",
|
||||
"install_fail": "Fail to install, please update manually",
|
||||
"rebooting": "Rebooting...",
|
||||
"reboot_fail": "Fail to reboot, please reboot manually",
|
||||
"custom_key_set": "Custom keybox set successfully",
|
||||
"custom_key_set_error": "Failed to set custom keybox",
|
||||
"no_file_selected": "No file selected"
|
||||
"no_file_selected": "No file selected",
|
||||
"system_app_not_found": "System app not found",
|
||||
"system_app_error": "Failed to add system app"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Security Patch",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Invalid boot format",
|
||||
"invalid_system": "Invalid system format",
|
||||
"invalid_vendor": "Invalid vendor format"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Add System App",
|
||||
"add": "Add",
|
||||
"current_list": "Current System App List"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Confirm Uninstall?",
|
||||
"uninstall_message": "Are you sure you want to uninstall Tricky Addon",
|
||||
"uninstall_cancel": "Cancel",
|
||||
"uninstall_confirm": "Confirm"
|
||||
}
|
||||
}
|
||||
@@ -1,7 +1,7 @@
|
||||
# Translation Guide
|
||||
## Fix Bad Translation
|
||||
1. Fork this repository.
|
||||
2. Find your language string file in `/module/webroot/locales/`.
|
||||
2. Find your language string file in `/module/webui/locales/`.
|
||||
3. Edit the string value with translated incorrectly.
|
||||
4. Create a Pull Request.
|
||||
|
||||
@@ -12,7 +12,8 @@
|
||||
|
||||
### Advanced
|
||||
1. Fork this repository.
|
||||
2. Make a copy of `/module/webroot/locales/A-template.json`
|
||||
2. Make a copy of `/module/webui/locales/A-template.json`
|
||||
3. Rename it to `language_code-COUNTRY_CODE.json`, e.g., `en-US.json`.
|
||||
4. Translate the string value inside.
|
||||
6. Create a Pull Request.
|
||||
5. Add the language code to `/module/webui/locales/available-lang.json`, this step is necessary for displaying the language in the WebUI.
|
||||
6. Create a Pull Request.
|
||||
|
||||
132
module/webui/locales/ar-EG.json
Normal file
132
module/webui/locales/ar-EG.json
Normal file
@@ -0,0 +1,132 @@
|
||||
{
|
||||
"language": "العربية",
|
||||
"system_default": "الإعدادات الافتراضية للنظام",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
"help": {
|
||||
"help_instructions": "تعليمات",
|
||||
"save_and_update": "حفظ",
|
||||
"save_and_update_description": "حفظ الإعدادات الحالية إلى target.txt.",
|
||||
"refresh": "تحديث",
|
||||
"refresh_description": "تحديث قائمة التطبيقات وقائمة الاستبعاد.",
|
||||
"select_deselect": "تحديد وإلغاء تحديد الكل",
|
||||
"select_description": "تحديد أو إلغاء تحديد جميع التطبيقات في الواجهة الحالية.",
|
||||
"select_denylist": "تحديد من قائمة الرفض",
|
||||
"select_denylist_description": "متاح في Magisk فقط، حدد التطبيقات الموجودة في قائمة الرفض. موصى به.",
|
||||
"deselect_unnecessary": "إلغاء تحديد غير الضروري",
|
||||
"deselect_unnecessary_description": "فئة غير ضرورية: وحدة Xposed، مدير الجذر، التطبيقات المتعلقة بالجذر، والتطبيقات العامة التي لا تتحقق أبدًا من حالة bootloader. يتطلب هذا الخيار اتصالاً بالإنترنت.",
|
||||
"add_system_app": "إضافة تطبيق نظام",
|
||||
"add_system_app_description": "إضافة تطبيق نظام محدد إلى قائمة التطبيقات.",
|
||||
"set_keybox": "تعيين AOSP و Keybox صالح",
|
||||
"set_keybox_description": "استبدال tricky store keybox.xml. سيتم استبدال keybox AOSP إذا لم يكن هناك keybox صالح آخر. يتطلب خيار keybox الصالح اتصالاً بالإنترنت.",
|
||||
"set_custom_keybox": "تعيين Keybox مخصص",
|
||||
"set_custom_keybox_description": "استيراد keybox من تخزين جهازك. يدعم فقط ملف xml.",
|
||||
"set_security_patch": "تعيين تصحيح الأمان",
|
||||
"set_security_patch_description": "تعيين تصحيح أمان مخصص. سيستخدم التكوين التلقائي تصحيح الأمان من وحدة PIF، مفعل بشكل افتراضي. اتركه فارغًا واحفظه لتعطيل التكوين التلقائي.",
|
||||
"set_verified_boot_hash": "تعيين تجزئة التمهيد الموثوق",
|
||||
"set_verified_boot_hash_description": "احصل على قيمة verifiedBootHash من عرض إثبات المفتاح. إصلاح حالة التمهيد غير الطبيعية عن طريق إعادة تعيين ro.boot.vbmeta.digest."
|
||||
},
|
||||
"update": {
|
||||
"update_available": "إصدار جديد جاهز",
|
||||
"redirect_to_release": "اضغط لتنزيل أحدث إصدار",
|
||||
"changelog": "سجل التغييرات",
|
||||
"install": "تثبيت",
|
||||
"reboot": "إعادة تشغيل"
|
||||
},
|
||||
"search_bar": {
|
||||
"search_placeholder": "بحث"
|
||||
},
|
||||
"functional_button": {
|
||||
"save_and_update_button": "حفظ",
|
||||
"uninstall_webui": "إلغاء تثبيت WebUI"
|
||||
},
|
||||
"loading": {
|
||||
"loading": "جارٍ التحميل..."
|
||||
},
|
||||
"menu": {
|
||||
"refresh": "تحديث",
|
||||
"select_all": "تحديد الكل",
|
||||
"deselect_all": "إلغاء تحديد الكل",
|
||||
"select_denylist": "تحديد من قائمة الرفض",
|
||||
"deselect_unnecessary": "إلغاء تحديد غير الضروري",
|
||||
"add_system_app": "إضافة تطبيق نظام",
|
||||
"set_aosp_keybox": "تعيين Keybox AOSP",
|
||||
"set_valid_keybox": "تعيين Keybox صالح",
|
||||
"set_custom_keybox": "تعيين Keybox مخصص",
|
||||
"set_verified_boot_hash": "تعيين تجزئة التمهيد الموثوق",
|
||||
"set_security_patch": "تعيين تصحيح الأمان",
|
||||
"about": "حول"
|
||||
},
|
||||
"boot_hash": {
|
||||
"title": "تجزئة التمهيد",
|
||||
"input_placeholder": "الصق تجزئة التمهيد الموثوق هنا",
|
||||
"save_button": "حفظ"
|
||||
},
|
||||
"about": {
|
||||
"module_name_line1": "Tricky Addon",
|
||||
"module_name_line2": "تحديث قائمة الأهداف",
|
||||
"by": "بواسطة",
|
||||
"telegram_channel": "قناة التليجرام",
|
||||
"github": "GitHub",
|
||||
"disclaimer": "هذه الوحدة ليست جزءًا من وحدة Tricky Store. لا تقم بالإبلاغ عن أي مشاكل إلى Tricky Store إذا واجهت أيًا منها.",
|
||||
"acknowledgment": "شكر وتقدير"
|
||||
},
|
||||
"prompt": {
|
||||
"no_internet": "يرجى التحقق من اتصالك بالإنترنت",
|
||||
"aosp_key_set": "تم تعيين keybox AOSP بنجاح",
|
||||
"key_set_error": "فشل في تحديث keybox",
|
||||
"valid_key_set": "تم تعيين keybox صالح بنجاح",
|
||||
"no_valid_fallback": "لم يتم العثور على keybox صالح، تم الاستبدال بـ keybox AOSP.",
|
||||
"boot_hash_set": "تم حفظ تجزئة التمهيد الموثوق بنجاح",
|
||||
"boot_hash_set_error": "فشل في تحديث تجزئة التمهيد الموثوق",
|
||||
"saved_target": "تم حفظ التكوين إلى target.txt",
|
||||
"save_error": "فشل في حفظ التكوين",
|
||||
"uninstall_prompt": "سيتم إزالة WebUI بعد إعادة التشغيل",
|
||||
"uninstall_failed": "فشل في إلغاء تثبيت WebUI",
|
||||
"new_update": "تحديث جديد متاح!",
|
||||
"downloading": "جارٍ تنزيل التحديث الجديد...",
|
||||
"downloaded": "اكتمل التنزيل",
|
||||
"download_fail": "فشل في تنزيل التحديث",
|
||||
"installing": "جارٍ تثبيت التحديث...",
|
||||
"installed": "تم التثبيت بنجاح، أعد التشغيل الآن.",
|
||||
"install_fail": "فشل في التثبيت، يرجى التحديث يدويًا",
|
||||
"rebooting": "جارٍ إعادة التشغيل...",
|
||||
"reboot_fail": "فشل في إعادة التشغيل، يرجى إعادة التشغيل يدويًا",
|
||||
"custom_key_set": "تم تعيين keybox مخصص بنجاح",
|
||||
"custom_key_set_error": "فشل في تعيين keybox مخصص",
|
||||
"no_file_selected": "لم يتم اختيار أي ملف",
|
||||
"system_app_not_found": "لم يتم العثور على تطبيق النظام",
|
||||
"system_app_error": "فشل في إضافة تطبيق النظام"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "تصحيح الأمان",
|
||||
"advanced_mode": "متقدم",
|
||||
"get_date": "الحصول على تاريخ تصحيح الأمان",
|
||||
"auto": "تلقائي",
|
||||
"save": "حفظ",
|
||||
"fetching": "جارٍ التحميل...",
|
||||
"fetched": "تم",
|
||||
"get_failed": "فشل في جلب تاريخ تصحيح الأمان",
|
||||
"auto_success": "تم تفعيل التكوين التلقائي بنجاح",
|
||||
"auto_failed": "فشل في تفعيل التكوين التلقائي",
|
||||
"save_success": "تم حفظ تصحيح الأمان بنجاح",
|
||||
"save_failed": "فشل في حفظ تصحيح الأمان",
|
||||
"value_empty": "تم تعطيل تكوين تصحيح الأمان",
|
||||
"invalid_all": "تنسيق غير صالح",
|
||||
"invalid_boot": "تنسيق تمهيد غير صالح",
|
||||
"invalid_system": "تنسيق نظام غير صالح",
|
||||
"invalid_vendor": "تنسيق بائع غير صالح"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "إضافة تطبيق نظام",
|
||||
"add": "إضافة",
|
||||
"current_list": "قائمة تطبيقات النظام الحالية"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "تأكيد إلغاء التثبيت؟",
|
||||
"uninstall_message": "هل أنت متأكد أنك تريد إلغاء تثبيت Tricky Addon",
|
||||
"uninstall_cancel": "إلغاء",
|
||||
"uninstall_confirm": "تأكيد"
|
||||
}
|
||||
}
|
||||
18
module/webui/locales/available-lang.json
Normal file
18
module/webui/locales/available-lang.json
Normal file
@@ -0,0 +1,18 @@
|
||||
{
|
||||
"languages": [
|
||||
"ar-EG",
|
||||
"en-US",
|
||||
"es-ES",
|
||||
"fr-FR",
|
||||
"id-ID",
|
||||
"it-IT",
|
||||
"ja-JP",
|
||||
"pl-PL",
|
||||
"ru-RU",
|
||||
"tl-PH",
|
||||
"tr-TR",
|
||||
"az-AZ",
|
||||
"zh-CN",
|
||||
"zh-TW"
|
||||
]
|
||||
}
|
||||
132
module/webui/locales/az-AZ.json
Normal file
132
module/webui/locales/az-AZ.json
Normal file
@@ -0,0 +1,132 @@
|
||||
{
|
||||
"language": "Azərbaycanca",
|
||||
"system_default": "Sistem Varsayılanı",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
"help": {
|
||||
"help_instructions": "Təlimatlar",
|
||||
"save_and_update": "Yadda saxla",
|
||||
"save_and_update_description": "Cari konfiqurasiyanı target.txt faylına yadda saxla.",
|
||||
"refresh": "Təzələ",
|
||||
"refresh_description": "Proqram siyahısını və istisna siyahısını təzələ.",
|
||||
"select_deselect": "Hamısını seç və seçimi ləğv et",
|
||||
"select_description": "Cari interfeysdəki bütün tətbiqləri seç və ya seçimdən çıxar.",
|
||||
"select_denylist": "DenyList-dən seç",
|
||||
"select_denylist_description": "Yalnız Magisk-də mövcuddur, DenyList-də olan tətbiqləri seçin. Tövsiyə olunur.",
|
||||
"deselect_unnecessary": "Lazımsızları seçmə",
|
||||
"deselect_unnecessary_description": "Lazımsız kateqoriya: Xposed modulu, kök meneceri, kök ilə əlaqəli tətbiqlər və heç vaxt bootloader statusunu yoxlamayan ümumi tətbiqlər. Bu seçim İnternet bağlantısı tələb edir.",
|
||||
"add_system_app": "Sistem Tətbiqi Əlavə et",
|
||||
"add_system_app_description": "Müəyyən bir sistem tətbiqini tətbiq siyahısına əlavə et.",
|
||||
"set_keybox": "AOSP & Etibarlı Keybox təyin et",
|
||||
"set_keybox_description": "Tricky store keybox.xml-ni dəyişdir. Əgər etibarlı keybox yoxdursa, AOSP keybox dəyişdiriləcək. Etibarlı keybox seçimi İnternet bağlantısı tələb edir.",
|
||||
"set_custom_keybox": "Özəl Keybox təyin et",
|
||||
"set_custom_keybox_description": "Keybox-u cihaz yaddaşından idxal et. Yalnız xml faylını dəstəkləyir.",
|
||||
"set_security_patch": "Təhlükəsizlik Təkmilləşdirməsini təyin et",
|
||||
"set_security_patch_description": "Xüsusi təhlükəsizlik təkmilləşdirməsini dəyişdir. Avtomatik konfiqurasiya, PIF modulundan təhlükəsizlik təkmilləşdirməsini istifadə edəcək, bu seçimi aktivdir. Avtomatik konfiqurasiyanı dayandırmaq üçün boş buraxın və yadda saxlayın.",
|
||||
"set_verified_boot_hash": "Təsdiqlənmiş Boot Hash təyin et",
|
||||
"set_verified_boot_hash_description": "Key Attestation Demo-dan verifiedBootHash dəyərini əldə et. ro.boot.vbmeta.digest-ni sıfırlayaraq qeyri-adi boot vəziyyətini düzəlt."
|
||||
},
|
||||
"update": {
|
||||
"update_available": "Yeni versiya mövcuddur",
|
||||
"redirect_to_release": "Ən son versiyanı yükləmək üçün klikləyin",
|
||||
"changelog": "Dəyişikliklər",
|
||||
"install": "Yüklə",
|
||||
"reboot": "Yenidən başladın"
|
||||
},
|
||||
"search_bar": {
|
||||
"search_placeholder": "Axtar"
|
||||
},
|
||||
"functional_button": {
|
||||
"save_and_update_button": "Yadda saxla",
|
||||
"uninstall_webui": "WebUI-ni sil"
|
||||
},
|
||||
"loading": {
|
||||
"loading": "Yüklənir..."
|
||||
},
|
||||
"menu": {
|
||||
"refresh": "Təzələ",
|
||||
"select_all": "Hamısını seç",
|
||||
"deselect_all": "Hamısının seçimini ləğv et",
|
||||
"select_denylist": "DenyList-dən seç",
|
||||
"deselect_unnecessary": "Lazımsızları seçmə",
|
||||
"add_system_app": "Sistem Tətbiqi Əlavə et",
|
||||
"set_aosp_keybox": "AOSP Keybox təyin et",
|
||||
"set_valid_keybox": "Etibarlı Keybox təyin et",
|
||||
"set_custom_keybox": "Özəl Keybox təyin et",
|
||||
"set_verified_boot_hash": "Təsdiqlənmiş Boot Hash təyin et",
|
||||
"set_security_patch": "Təhlükəsizlik Təkmilləşdirməsini təyin et",
|
||||
"about": "Haqqında"
|
||||
},
|
||||
"boot_hash": {
|
||||
"title": "Boot Hash",
|
||||
"input_placeholder": "Təsdiqlənmiş Boot Hash-ı buraya yapışdırın",
|
||||
"save_button": "Yadda saxla"
|
||||
},
|
||||
"about": {
|
||||
"module_name_line1": "Tricky Addon",
|
||||
"module_name_line2": "Hədəf Siyahısını Yenilə",
|
||||
"by": "tərəfindən",
|
||||
"telegram_channel": "Telegram Kanalı",
|
||||
"github": "GitHub",
|
||||
"disclaimer": "Bu mod, Tricky Store modulunun bir hissəsi deyil. Qarşılaşılan hər hansı bir problemi Tricky Store-a bildirməyin.",
|
||||
"acknowledgment": "Təşəkkür"
|
||||
},
|
||||
"prompt": {
|
||||
"no_internet": "İnternet bağlantınızı yoxlayın",
|
||||
"aosp_key_set": "AOSP keybox uğurla təyin olundu",
|
||||
"key_set_error": "Keybox-u yeniləmək mümkün olmadı",
|
||||
"valid_key_set": "Etibarlı keybox uğurla təyin olundu",
|
||||
"no_valid_fallback": "Etibarlı keybox tapılmadı, AOSP keybox ilə dəyişdirildi.",
|
||||
"boot_hash_set": "Təsdiqlənmiş Boot Hash uğurla yadda saxlanıldı",
|
||||
"boot_hash_set_error": "Təsdiqlənmiş Boot Hash yenilənə bilmədi",
|
||||
"saved_target": "Konfiqurasiya target.txt-yə yadda saxlanıldı",
|
||||
"save_error": "Konfiqurasiyanı yadda saxlamaq mümkün olmadı",
|
||||
"uninstall_prompt": "WebUI yenidən başladıqdan sonra silinəcək",
|
||||
"uninstall_failed": "WebUI-ni silmək mümkün olmadı",
|
||||
"new_update": "Yeni bir yeniləmə mövcuddur!",
|
||||
"downloading": "Yeni yeniləməni yükləyir...",
|
||||
"downloaded": "Yükləmə tamamlandı",
|
||||
"download_fail": "Yeniləməni yükləmək mümkün olmadı",
|
||||
"installing": "Yeniləmə quraşdırılır...",
|
||||
"installed": "Uğurla quraşdırıldı, indi yenidən başladın.",
|
||||
"install_fail": "Quraşdırmaq mümkün olmadı, zəhmət olmasa əl ilə yeniləyin",
|
||||
"rebooting": "Yenidən başladılır...",
|
||||
"reboot_fail": "Yenidən başlatmaq mümkün olmadı, zəhmət olmasa əl ilə yenidən başlayın",
|
||||
"custom_key_set": "Özəl keybox uğurla təyin olundu",
|
||||
"custom_key_set_error": "Özəl keybox təyin oluna bilmədi",
|
||||
"no_file_selected": "Fayl seçilməyib",
|
||||
"system_app_not_found": "Sistem tətbiqi tapılmadı",
|
||||
"system_app_error": "Sistem tətbiqini əlavə etmək mümkün olmadı"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Təhlükəsizlik Təkmilləşdirməsi",
|
||||
"advanced_mode": "Ətraflı",
|
||||
"get_date": "Təhlükəsizlik Təkmilləşdirməsinin Tarixini Al",
|
||||
"auto": "Avtomatik",
|
||||
"save": "Yadda saxla",
|
||||
"fetching": "Yığılır...",
|
||||
"fetched": "Tamam",
|
||||
"get_failed": "Təhlükəsizlik təkmilləşdirməsinin tarixini almaq mümkün olmadı",
|
||||
"auto_success": "Avtomatik konfiqurasiya uğurla aktivləşdirildi",
|
||||
"auto_failed": "Avtomatik konfiqurasiyanı aktivləşdirmək mümkün olmadı",
|
||||
"save_success": "Təhlükəsizlik təkmilləşdirməsi uğurla yadda saxlanıldı",
|
||||
"save_failed": "Təhlükəsizlik təkmilləşdirməsini yadda saxlamaq mümkün olmadı",
|
||||
"value_empty": "Təhlükəsizlik təkmilləşdirməsi konfiqurasiyası deaktivdir",
|
||||
"invalid_all": "Yanlış format",
|
||||
"invalid_boot": "Yanlış boot formatı",
|
||||
"invalid_system": "Yanlış sistem formatı",
|
||||
"invalid_vendor": "Yanlış vendor formatı"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Sistem Tətbiqi Əlavə et",
|
||||
"add": "Əlavə et",
|
||||
"current_list": "Cari Sistem Tətbiqi Siyahısı"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Silinməni Təsdiqləyirsiniz?",
|
||||
"uninstall_message": "Tricky Addon-u silmək istədiyinizə əminsinizmi?",
|
||||
"uninstall_cancel": "İmtina et",
|
||||
"uninstall_confirm": "Təsdiqlə"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "English",
|
||||
"system_default": "System Default",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Available in Magisk only, select apps that are in the DenyList. Recommended.",
|
||||
"deselect_unnecessary": "Deselect Unnecessary",
|
||||
"deselect_unnecessary_description": "Unnecessary category: Xposed module, root manager, root-related apps, and general apps that never check bootloader status. This option requires Internet connection.",
|
||||
"add_system_app": "Add System App",
|
||||
"add_system_app_description": "Add specific system app to app list.",
|
||||
"set_keybox": "Set AOSP & Valid Keybox",
|
||||
"set_keybox_description": "Replace tricky store keybox.xml. AOSP keybox will be replaced if there's no more valid keybox. Valid keybox option requires Internet connection.",
|
||||
"set_custom_keybox": "Set Custom Keybox",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Deselect All",
|
||||
"select_denylist": "Select From DenyList",
|
||||
"deselect_unnecessary": "Deselect Unnecessary",
|
||||
"add_system_app": "Add System App",
|
||||
"set_aosp_keybox": "Set AOSP Keybox",
|
||||
"set_valid_keybox": "Set Valid Keybox",
|
||||
"set_custom_keybox": "Set Custom Keybox",
|
||||
@@ -86,12 +90,14 @@
|
||||
"download_fail": "Fail to download update",
|
||||
"installing": "Installing update...",
|
||||
"installed": "Installed successfully, reboot now.",
|
||||
"install_fail": "Fail to install, please update manual",
|
||||
"install_fail": "Fail to install, please update manually",
|
||||
"rebooting": "Rebooting...",
|
||||
"reboot_fail": "Fail to reboot, please reboot manually",
|
||||
"custom_key_set": "Custom keybox set successfully",
|
||||
"custom_key_set_error": "Failed to set custom keybox",
|
||||
"no_file_selected": "No file selected"
|
||||
"no_file_selected": "No file selected",
|
||||
"system_app_not_found": "System app not found",
|
||||
"system_app_error": "Failed to add system app"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Security Patch",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Invalid boot format",
|
||||
"invalid_system": "Invalid system format",
|
||||
"invalid_vendor": "Invalid vendor format"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Add System App",
|
||||
"add": "Add",
|
||||
"current_list": "Current System App List"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Confirm Uninstall?",
|
||||
"uninstall_message": "Are you sure you want to uninstall Tricky Addon",
|
||||
"uninstall_cancel": "Cancel",
|
||||
"uninstall_confirm": "Confirm"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "Español",
|
||||
"system_default": "Predeterminado del sistema",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Disponible solo en Magisk, selecciona aplicaciones que están en la DenyList. Recomendado.",
|
||||
"deselect_unnecessary": "Deseleccionar innecesarios",
|
||||
"deselect_unnecessary_description": "Categorías innecesarias: módulos Xposed, gestores de root, aplicaciones relacionadas con root y aplicaciones generales que nunca verifican el estado del bootloader. Esta opción requiere conexión a Internet.",
|
||||
"add_system_app": "Agregar Aplicación del Sistema",
|
||||
"add_system_app_description": "Agregar una aplicación del sistema específica a la lista de aplicaciones.",
|
||||
"set_keybox": "Configurar AOSP y Keybox Válido",
|
||||
"set_keybox_description": "Reemplazar el archivo keybox.xml de Tricky Store. El AOSP Keybox será reemplazado si no hay un keybox válido. Esta opción requiere conexión a Internet.",
|
||||
"set_custom_keybox": "Establecer Keybox Personalizado",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Deseleccionar Todo",
|
||||
"select_denylist": "Seleccionar desde DenyList",
|
||||
"deselect_unnecessary": "Deseleccionar innecesarios",
|
||||
"add_system_app": "Agregar Aplicación del Sistema",
|
||||
"set_aosp_keybox": "Configurar AOSP Keybox",
|
||||
"set_valid_keybox": "Configurar Keybox Válido",
|
||||
"set_custom_keybox": "Establecer Keybox Personalizado",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "Error al reiniciar, reinicia manualmente",
|
||||
"custom_key_set": "Keybox personalizado establecido con éxito",
|
||||
"custom_key_set_error": "Error al establecer el keybox personalizado",
|
||||
"no_file_selected": "Ningún archivo seleccionado"
|
||||
"no_file_selected": "Ningún archivo seleccionado",
|
||||
"system_app_not_found": "Aplicación del sistema no encontrada",
|
||||
"system_app_error": "Error al agregar la aplicación del sistema"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Parche de Seguridad",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Formato de boot inválido",
|
||||
"invalid_system": "Formato de system inválido",
|
||||
"invalid_vendor": "Formato de vendor inválido"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Añadir aplicación del sistema",
|
||||
"add": "Añadir",
|
||||
"current_list": "Lista actual de aplicaciones del sistema"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "¿Confirmar desinstalación?",
|
||||
"uninstall_message": "¿Está seguro de que desea desinstalar Tricky Addon",
|
||||
"uninstall_cancel": "Cancelar",
|
||||
"uninstall_confirm": "Confirmar"
|
||||
}
|
||||
}
|
||||
132
module/webui/locales/fr-FR.json
Normal file
132
module/webui/locales/fr-FR.json
Normal file
@@ -0,0 +1,132 @@
|
||||
{
|
||||
"language": "Français",
|
||||
"system_default": "Langage système",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
"help": {
|
||||
"help_instructions": "Instructions",
|
||||
"save_and_update": "Enregistrer",
|
||||
"save_and_update_description": "Enregistrer la configuration actuelle dans target.txt.",
|
||||
"refresh": "Actualiser",
|
||||
"refresh_description": "Actualiser la liste des applications et la liste d'exclusions.",
|
||||
"select_deselect": "Tout sélectionner & désélectionner",
|
||||
"select_description": "Sélectionner ou désélectionner toutes les applications de l'interface actuelle.",
|
||||
"select_DenyList": "Sélectionner depuis la DenyList",
|
||||
"select_DenyList_description": "Disponible uniquement sur Magisk, sélectionner les applications présentes dans la DenyList. Recommandé.",
|
||||
"deselect_unnecessary": "Désélectionner les applications inutiles",
|
||||
"deselect_unnecessary_description": "Catégorie inutile : module Xposed, gestionnaire root, applications liées au root et applications générales ne vérifiant jamais l'état du bootloader. Cette option nécessite une connexion Internet.",
|
||||
"add_system_app": "Ajouter une application système",
|
||||
"add_system_app_description": "Ajouter une application système spécifique à la liste.",
|
||||
"set_keybox": "Définir une Keybox AOSP & Valide",
|
||||
"set_keybox_description": "Remplacer le fichier keybox.xml de Tricky Store. La keybox AOSP sera remplacée s'il n'y a plus de keybox valide. L'option keybox valide nécessite une connexion Internet.",
|
||||
"set_custom_keybox": "Définir une Keybox personnalisée",
|
||||
"set_custom_keybox_description": "Importer une keybox depuis le stockage de votre appareil. Seuls les fichiers XML sont pris en charge.",
|
||||
"set_security_patch": "Définir le patch de sécurité",
|
||||
"set_security_patch_description": "Définir un patch de sécurité personnalisé. La configuration automatique utilisera le patch de sécurité du module PIF, activée par défaut. Laisser vide et enregistrer pour désactiver la configuration automatique.",
|
||||
"set_verified_boot_hash": "Définir le hash Verified Boot",
|
||||
"set_verified_boot_hash_description": "Obtenir la valeur verifiedBootHash depuis Key Attestation Demo. Corriger un état de démarrage anormal en réinitialisant ro.boot.vbmeta.digest."
|
||||
},
|
||||
"update": {
|
||||
"update_available": "Une nouvelle version est disponible",
|
||||
"redirect_to_release": "appuyer pour télécharger la dernière version",
|
||||
"changelog": "Journal des modifications",
|
||||
"install": "Installer",
|
||||
"reboot": "Redémarrer"
|
||||
},
|
||||
"search_bar": {
|
||||
"search_placeholder": "Rechercher"
|
||||
},
|
||||
"functional_button": {
|
||||
"save_and_update_button": "Enregistrer",
|
||||
"uninstall_webui": "Désinstaller la WebUI"
|
||||
},
|
||||
"loading": {
|
||||
"loading": "Chargement…"
|
||||
},
|
||||
"menu": {
|
||||
"refresh": "Actualiser",
|
||||
"select_all": "Tout sélectionner",
|
||||
"deselect_all": "Tout désélectionner",
|
||||
"select_DenyList": "Sélectionner depuis la DenyList",
|
||||
"deselect_unnecessary": "Désélectionner les applications inutiles",
|
||||
"add_system_app": "Ajouter une application système",
|
||||
"set_aosp_keybox": "Définir une Keybox AOSP",
|
||||
"set_valid_keybox": "Définir une Keybox valide",
|
||||
"set_custom_keybox": "Définir une Keybox personnalisée",
|
||||
"set_verified_boot_hash": "Définir le hash Verified Boot",
|
||||
"set_security_patch": "Définir le patch de sécurité",
|
||||
"about": "À propos"
|
||||
},
|
||||
"boot_hash": {
|
||||
"title": "Hash de démarrage",
|
||||
"input_placeholder": "Collez votre hash Verified Boot ici",
|
||||
"save_button": "Enregistrer"
|
||||
},
|
||||
"about": {
|
||||
"module_name_line1": "Tricky Addon",
|
||||
"module_name_line2": "Mettre à jour la liste cible",
|
||||
"by": "par",
|
||||
"telegram_channel": "Canal Telegram",
|
||||
"github": "GitHub",
|
||||
"disclaimer": "Ce module ne fait pas partie du module Tricky Store. NE signalez PAS de problèmes à Tricky Store en cas d'erreur.",
|
||||
"acknowledgment": "Remerciements"
|
||||
},
|
||||
"prompt": {
|
||||
"no_internet": "Veuillez vérifier votre connexion Internet",
|
||||
"aosp_key_set": "Keybox AOSP définie avec succès",
|
||||
"key_set_error": "Échec de la mise à jour de la Keybox",
|
||||
"valid_key_set": "Keybox valide définie avec succès",
|
||||
"no_valid_fallback": "Aucune Keybox valide trouvée, remplacée par le Keybox AOSP.",
|
||||
"boot_hash_set": "Hash Verified Boot enregistré avec succès",
|
||||
"boot_hash_set_error": "Échec de la mise à jour du hash Verified Boot",
|
||||
"saved_target": "Configuration enregistrée dans target.txt",
|
||||
"save_error": "Échec de l'enregistrement de la configuration",
|
||||
"uninstall_prompt": "La WebUI sera supprimée après le redémarrage",
|
||||
"uninstall_failed": "Échec de la désinstallation de la WebUI",
|
||||
"new_update": "Une nouvelle mise à jour est disponible !",
|
||||
"downloading": "Téléchargement de la mise à jour…",
|
||||
"downloaded": "Téléchargement terminé",
|
||||
"download_fail": "Échec du téléchargement de la mise à jour",
|
||||
"installing": "Installation de la mise à jour…",
|
||||
"installed": "Installation réussie, redémarrez maintenant.",
|
||||
"install_fail": "Échec de l'installation, veuillez mettre à jour manuellement",
|
||||
"rebooting": "Redémarrage…",
|
||||
"reboot_fail": "Échec du redémarrage, veuillez redémarrer manuellement",
|
||||
"custom_key_set": "Keybox personnalisée définie avec succès",
|
||||
"custom_key_set_error": "Échec de la définition de la Keybox personnalisée",
|
||||
"no_file_selected": "Aucun fichier sélectionné",
|
||||
"system_app_not_found": "Application système introuvable",
|
||||
"system_app_error": "Échec de l'ajout de l'application système"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Patch de sécurité",
|
||||
"advanced_mode": "Avancé",
|
||||
"get_date": "Obtenir la date du patch de sécurité",
|
||||
"auto": "Automatique",
|
||||
"save": "Enregistrer",
|
||||
"fetching": "Récupération…",
|
||||
"fetched": "Terminé",
|
||||
"get_failed": "Échec de la récupération de la date du patch de sécurité",
|
||||
"auto_success": "Configuration automatique activée avec succès",
|
||||
"auto_failed": "Échec de l'activation de la configuration automatique",
|
||||
"save_success": "Patch de sécurité enregistré avec succès",
|
||||
"save_failed": "Échec de l'enregistrement du patch de sécurité",
|
||||
"value_empty": "Configuration du patch de sécurité désactivée",
|
||||
"invalid_all": "Format invalide",
|
||||
"invalid_boot": "Format de démarrage invalide",
|
||||
"invalid_system": "Format du système invalide",
|
||||
"invalid_vendor": "Format du fournisseur invalide"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Ajouter une application système",
|
||||
"add": "Ajouter",
|
||||
"current_list": "Liste des applications système actuelles"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Confirmer la désinstallation ?",
|
||||
"uninstall_message": "Êtes-vous sûr(e) de vouloir désinstaller Tricky Addon",
|
||||
"uninstall_cancel": "Annuler",
|
||||
"uninstall_confirm": "Confirmer"
|
||||
}
|
||||
}
|
||||
132
module/webui/locales/id-ID.json
Normal file
132
module/webui/locales/id-ID.json
Normal file
@@ -0,0 +1,132 @@
|
||||
{
|
||||
"language": "Bahasa Indonesia",
|
||||
"system_default": "Default Sistem",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
"help": {
|
||||
"help_instructions": "Panduan",
|
||||
"save_and_update": "Simpan",
|
||||
"save_and_update_description": "Simpan konfigurasi saat ini ke target.txt.",
|
||||
"refresh": "Segarkan",
|
||||
"refresh_description": "Perbarui daftar aplikasi dan daftar pengecualian.",
|
||||
"select_deselect": "Pilih & Batalkan Pilihan Semua",
|
||||
"select_description": "Pilih atau batalkan pilihan semua aplikasi yang ditampilkan.",
|
||||
"select_denylist": "Pilih dari Denylist",
|
||||
"select_denylist_description": "Hanya untuk Magisk, pilih aplikasi yang ada di Denylist. Disarankan.",
|
||||
"deselect_unnecessary": "Batalkan Pilihan yang Tidak Perlu",
|
||||
"deselect_unnecessary_description": "Kategori tidak perlu: Modul Xposed, pengelola root, aplikasi terkait root, dan aplikasi yang tidak pernah memeriksa status bootloader. Opsi ini memerlukan koneksi internet.",
|
||||
"add_system_app": "Tambahkan Aplikasi Sistem",
|
||||
"add_system_app_description": "Tambahkan aplikasi sistem tertentu ke daftar aplikasi.",
|
||||
"set_keybox": "Ganti Keybox AOSP & Valid",
|
||||
"set_keybox_description": "Ganti file keybox.xml bawaan. Jika tidak ada keybox valid, maka keybox AOSP akan digunakan. Opsi ini memerlukan koneksi internet.",
|
||||
"set_custom_keybox": "Gunakan Keybox Kustom",
|
||||
"set_custom_keybox_description": "Impor keybox dari penyimpanan perangkat. Hanya mendukung file XML.",
|
||||
"set_security_patch": "Atur Patch Keamanan",
|
||||
"set_security_patch_description": "Sesuaikan patch keamanan. Konfigurasi otomatis akan menggunakan patch dari modul PIF dan aktif secara default. Kosongkan dan simpan untuk menonaktifkan konfigurasi otomatis.",
|
||||
"set_verified_boot_hash": "Atur Verified Boot Hash",
|
||||
"set_verified_boot_hash_description": "Ambil nilai verifiedBootHash dari aplikasi Key Attestation Demo. Perbaiki status boot yang bermasalah dengan mereset ro.boot.vbmeta.digest."
|
||||
},
|
||||
"update": {
|
||||
"update_available": "Versi baru tersedia!",
|
||||
"redirect_to_release": "Ketuk untuk mengunduh versi terbaru",
|
||||
"changelog": "Catatan Perubahan",
|
||||
"install": "Pasang",
|
||||
"reboot": "Mulai Ulang"
|
||||
},
|
||||
"search_bar": {
|
||||
"search_placeholder": "Cari"
|
||||
},
|
||||
"functional_button": {
|
||||
"save_and_update_button": "Simpan",
|
||||
"uninstall_webui": "Copot Pemasangan WebUI"
|
||||
},
|
||||
"loading": {
|
||||
"loading": "Memuat..."
|
||||
},
|
||||
"menu": {
|
||||
"refresh": "Segarkan",
|
||||
"select_all": "Pilih Semua",
|
||||
"deselect_all": "Batalkan Semua Pilihan",
|
||||
"select_denylist": "Pilih dari Denylist",
|
||||
"deselect_unnecessary": "Batalkan Pilihan yang Tidak Perlu",
|
||||
"add_system_app": "Tambahkan Aplikasi Sistem",
|
||||
"set_aosp_keybox": "Gunakan Keybox AOSP",
|
||||
"set_valid_keybox": "Gunakan Keybox Valid",
|
||||
"set_custom_keybox": "Gunakan Keybox Kustom",
|
||||
"set_verified_boot_hash": "Atur Verified Boot Hash",
|
||||
"set_security_patch": "Atur Patch Keamanan",
|
||||
"about": "Tentang"
|
||||
},
|
||||
"boot_hash": {
|
||||
"title": "Verified Boot Hash",
|
||||
"input_placeholder": "Tempel Verified Boot Hash Anda di sini",
|
||||
"save_button": "Simpan"
|
||||
},
|
||||
"about": {
|
||||
"module_name_line1": "Tricky Addon",
|
||||
"module_name_line2": "Perbarui Daftar Target",
|
||||
"by": "oleh",
|
||||
"telegram_channel": "Saluran Telegram",
|
||||
"github": "GitHub",
|
||||
"disclaimer": "Modul ini bukan bagian dari Tricky Store. Jangan laporkan masalah ke Tricky Store jika mengalami kendala.",
|
||||
"acknowledgment": "Pengakuan"
|
||||
},
|
||||
"prompt": {
|
||||
"no_internet": "Periksa koneksi internet Anda",
|
||||
"aosp_key_set": "Keybox AOSP berhasil digunakan",
|
||||
"key_set_error": "Gagal memperbarui keybox",
|
||||
"valid_key_set": "Keybox valid berhasil digunakan",
|
||||
"no_valid_fallback": "Tidak ada keybox valid, menggunakan keybox AOSP.",
|
||||
"boot_hash_set": "Verified Boot Hash berhasil disimpan",
|
||||
"boot_hash_set_error": "Gagal memperbarui Verified Boot Hash",
|
||||
"saved_target": "Konfigurasi disimpan ke target.txt",
|
||||
"save_error": "Gagal menyimpan konfigurasi",
|
||||
"uninstall_prompt": "WebUI akan dicopot setelah perangkat dimulai ulang",
|
||||
"uninstall_failed": "Gagal mencopot pemasangan WebUI",
|
||||
"new_update": "Pembaruan baru tersedia!",
|
||||
"downloading": "Mengunduh pembaruan...",
|
||||
"downloaded": "Unduhan selesai",
|
||||
"download_fail": "Gagal mengunduh pembaruan",
|
||||
"installing": "Memasang pembaruan...",
|
||||
"installed": "Pembaruan berhasil dipasang, mulai ulang sekarang.",
|
||||
"install_fail": "Gagal memasang, silakan perbarui secara manual",
|
||||
"rebooting": "Memulai ulang...",
|
||||
"reboot_fail": "Gagal memulai ulang, silakan lakukan secara manual",
|
||||
"custom_key_set": "Keybox kustom berhasil digunakan",
|
||||
"custom_key_set_error": "Gagal menggunakan keybox kustom",
|
||||
"no_file_selected": "Tidak ada file yang dipilih",
|
||||
"system_app_not_found": "Aplikasi sistem tidak ditemukan",
|
||||
"system_app_error": "Gagal menambahkan aplikasi sistem"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Patch Keamanan",
|
||||
"advanced_mode": "Mode Lanjutan",
|
||||
"get_date": "Ambil Tanggal Patch Keamanan",
|
||||
"auto": "Otomatis",
|
||||
"save": "Simpan",
|
||||
"fetching": "Mengambil...",
|
||||
"fetched": "Selesai",
|
||||
"get_failed": "Gagal mengambil tanggal patch keamanan",
|
||||
"auto_success": "Konfigurasi otomatis berhasil diaktifkan",
|
||||
"auto_failed": "Gagal mengaktifkan konfigurasi otomatis",
|
||||
"save_success": "Patch keamanan berhasil disimpan",
|
||||
"save_failed": "Gagal menyimpan patch keamanan",
|
||||
"value_empty": "Konfigurasi patch keamanan dinonaktifkan",
|
||||
"invalid_all": "Format tidak valid",
|
||||
"invalid_boot": "Format boot tidak valid",
|
||||
"invalid_system": "Format sistem tidak valid",
|
||||
"invalid_vendor": "Format vendor tidak valid"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Tambah Aplikasi Sistem",
|
||||
"add": "Tambah",
|
||||
"current_list": "Daftar Aplikasi Sistem Saat Ini"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Konfirmasi Hapus?",
|
||||
"uninstall_message": "Apakah Anda yakin ingin menghapus Tricky Addon",
|
||||
"uninstall_cancel": "Batal",
|
||||
"uninstall_confirm": "Konfirmasi"
|
||||
}
|
||||
}
|
||||
132
module/webui/locales/it-IT.json
Normal file
132
module/webui/locales/it-IT.json
Normal file
@@ -0,0 +1,132 @@
|
||||
{
|
||||
"language": "Italiano",
|
||||
"system_default": "Predefinito di sistema",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
"help": {
|
||||
"help_instructions": "Istruzioni",
|
||||
"save_and_update": "Salva",
|
||||
"save_and_update_description": "Salva la configurazione corrente in target.txt.",
|
||||
"refresh": "Aggiorna",
|
||||
"refresh_description": "Aggiorna l'elenco delle app e l'elenco delle esclusioni.",
|
||||
"select_deselect": "Seleziona & Deseleziona Tutto",
|
||||
"select_description": "Seleziona o deseleziona tutte le app nell'interfaccia corrente.",
|
||||
"select_denylist": "Seleziona da DenyList",
|
||||
"select_denylist_description": "Disponibile solo in Magisk, seleziona le app presenti nella DenyList. Consigliato.",
|
||||
"deselect_unnecessary": "Deseleziona Non necessari",
|
||||
"deselect_unnecessary_description": "Categoria non necessaria: modulo Xposed, root manager, app correlate al root e app generali che non controllano mai lo stato del bootloader. Questa opzione richiede una connessione a Internet.",
|
||||
"add_system_app": "Aggiungi App di Sistema",
|
||||
"add_system_app_description": "Aggiungi app di sistema specifiche all'elenco delle app.",
|
||||
"set_keybox": "Imposta AOSP & Keybox Valida",
|
||||
"set_keybox_description": "Sostituisci tricky store keybox.xml. La keybox AOSP verrà sostituita se non ci sono più keybox valide. L'opzione valida per la keybox richiede una connessione Internet.",
|
||||
"set_custom_keybox": "Imposta Keybox Personalizzata",
|
||||
"set_custom_keybox_description": "Importa keybox dalla memoria del tuo dispositivo. Supporta solo file xml.",
|
||||
"set_security_patch": "Imposta Patch di Sicurezza",
|
||||
"set_security_patch_description": "Imposta spoofing patch di sicurezza personalizzato. La configurazione automatica utilizzerà la patch di sicurezza dal modulo PIF, abilitato di default. Lascia vuoto e salva per disabilitare la configurazione automatica.",
|
||||
"set_verified_boot_hash": "Imposta Boot Hash Verificato",
|
||||
"set_verified_boot_hash_description": "Ottieni il valore verifiedBootHash da Key Attestation Demo. Correggi lo stato di avvio anomalo reimpostando ro.boot.vbmeta.digest."
|
||||
},
|
||||
"update": {
|
||||
"update_available": "È pronta una nuova versione",
|
||||
"redirect_to_release": "tocca per scaricare l'ultima versione",
|
||||
"changelog": "Changelog",
|
||||
"install": "Installa",
|
||||
"reboot": "Riavvio"
|
||||
},
|
||||
"search_bar": {
|
||||
"search_placeholder": "Ricerca"
|
||||
},
|
||||
"functional_button": {
|
||||
"save_and_update_button": "Salva",
|
||||
"uninstall_webui": "Disinstalla WebUI"
|
||||
},
|
||||
"loading": {
|
||||
"loading": "Caricamento..."
|
||||
},
|
||||
"menu": {
|
||||
"refresh": "Aggiorna",
|
||||
"select_all": "Seleziona Tutto",
|
||||
"deselect_all": "Deseleziona tutto",
|
||||
"select_denylist": "Seleziona da DenyList",
|
||||
"deselect_unnecessary": "Deseleziona Non necessari",
|
||||
"add_system_app": "Aggiungi App di Sistema",
|
||||
"set_aosp_keybox": "Imposta Keybox AOSP",
|
||||
"set_valid_keybox": "Imposta Keybox Valida",
|
||||
"set_custom_keybox": "Imposta Keybox Personalizzata",
|
||||
"set_verified_boot_hash": "Imposta Boot Hash Verificato",
|
||||
"set_security_patch": "Imposta Patch di Sicurezza",
|
||||
"about": "About"
|
||||
},
|
||||
"boot_hash": {
|
||||
"title": "Boot Hash",
|
||||
"input_placeholder": "Incolla qui il tuo Boot Hash verificato",
|
||||
"save_button": "Salva"
|
||||
},
|
||||
"about": {
|
||||
"module_name_line1": "Tricky Addon",
|
||||
"module_name_line2": "Aggiorna Target List",
|
||||
"by": "by",
|
||||
"telegram_channel": "Canale Telegram",
|
||||
"github": "GitHub",
|
||||
"disclaimer": "Questo modulo non fa parte del modulo Tricky Store. NON segnalare eventuali problemi a Tricky Store se riscontrati.",
|
||||
"acknowledgment": "Riconoscimento"
|
||||
},
|
||||
"prompt": {
|
||||
"no_internet": "Controlla la tua connessione Internet",
|
||||
"aosp_key_set": "Keybox AOSP impostata correttamente",
|
||||
"key_set_error": "Impossibile aggiornare keybox",
|
||||
"valid_key_set": "Keybox valida impostata correttamente",
|
||||
"no_valid_fallback": "Nessuna keybox valida trovata, sostituito con keybox AOSP.",
|
||||
"boot_hash_set": "Boot Hash Verificato salvato correttamente",
|
||||
"boot_hash_set_error": "Impossibile aggiornare Boot Hash Verificato",
|
||||
"saved_target": "Configurazione salvata in target.txt",
|
||||
"save_error": "Impossibile salvare la configurazione",
|
||||
"uninstall_prompt": "WebUI verrà rimosso dopo il riavvio",
|
||||
"uninstall_failed": "Impossibile disinstallare WebUI",
|
||||
"new_update": "È disponibile un nuovo aggiornamento!",
|
||||
"downloading": "Download nuovo aggiornamento...",
|
||||
"downloaded": "Download completato",
|
||||
"download_fail": "Impossibile scaricare l'aggiornamento",
|
||||
"installing": "Installazione aggiornamento...",
|
||||
"installed": "Installato correttamente, riavvia ora.",
|
||||
"install_fail": "Installazione non riuscita, aggiorna manualmente",
|
||||
"rebooting": "Riavvio...",
|
||||
"reboot_fail": "Impossibile riavviare, riavvia manualmente",
|
||||
"custom_key_set": "Keybox personalizzata impostata correttamente",
|
||||
"custom_key_set_error": "Impossibile impostare keybox personalizzata",
|
||||
"no_file_selected": "Nessun file selezionato",
|
||||
"system_app_not_found": "App di sistema non trovata",
|
||||
"system_app_error": "Impossibile aggiungere l'app di sistema"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Patch di sicurezza",
|
||||
"advanced_mode": "Avanzato",
|
||||
"get_date": "Ottieni data patch di sicurezza",
|
||||
"auto": "Auto",
|
||||
"save": "Salva",
|
||||
"fetching": "Recupero...",
|
||||
"fetched": "Fatto",
|
||||
"get_failed": "Impossibile recuperare la data della patch di sicurezza",
|
||||
"auto_success": "Configurazione automatica abilitata correttamente",
|
||||
"auto_failed": "Impossibile abilitare la configurazione automatica",
|
||||
"save_success": "Patch di sicurezza salvata correttamente",
|
||||
"save_failed": "Impossibile salvare la patch di sicurezza",
|
||||
"value_empty": "La configurazione della patch di sicurezza è disabilitata",
|
||||
"invalid_all": "Formato non valido",
|
||||
"invalid_boot": "Formato boot non valido",
|
||||
"invalid_system": "Formato system non valido",
|
||||
"invalid_vendor": "Formato vendor non valido"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Aggiungi app di sistema",
|
||||
"add": "Aggiungi",
|
||||
"current_list": "Elenco attuale delle app di sistema"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Confermi disinstallazione?",
|
||||
"uninstall_message": "Sei sicuro di voler disinstallare Tricky Addon",
|
||||
"uninstall_cancel": "Annulla",
|
||||
"uninstall_confirm": "Conferma"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "日本語",
|
||||
"system_default": "システムデフォルト",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Magisk の環境でのみ使用可能です。Deny List 内のアプリを選択します(推奨)。",
|
||||
"deselect_unnecessary": "不要な選択を解除",
|
||||
"deselect_unnecessary_description": "不要なカテゴリー: Xposed モジュール、root マネージャー、root 関連アプリ、Bootloader の状態を確認しない一般的なアプリです。このオプションはインターネット接続が必要です。",
|
||||
"add_system_app": "システムアプリを追加",
|
||||
"add_system_app_description": "特定のシステムアプリをアプリリストに追加します。",
|
||||
"set_keybox": "AOSP と 有効な Keybox",
|
||||
"set_keybox_description": "Tricky Store の keybox.xml を置き換えます。有効な Keybox がなくなった場合は、AOSP Keybox に置き換えられます。インターネット接続が必要です。",
|
||||
"set_custom_keybox": "カスタム Keybox を設定",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "すべての選択を解除",
|
||||
"select_denylist": "DenyList から選択",
|
||||
"deselect_unnecessary": "不要な選択を解除",
|
||||
"add_system_app": "システムアプリを追加",
|
||||
"set_aosp_keybox": "AOSP Keybox を設定",
|
||||
"set_valid_keybox": "有効な Keybox を設定",
|
||||
"set_custom_keybox": "カスタム Keybox を設定",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "再起動に失敗しました。手動で再起動してください。",
|
||||
"custom_key_set": "カスタム Keybox の設定に成功しました",
|
||||
"custom_key_set_error": "カスタム Keybox の設定に失敗しました",
|
||||
"no_file_selected": "ファイルが選択されていません"
|
||||
"no_file_selected": "ファイルが選択されていません",
|
||||
"system_app_not_found": "システムアプリが見つかりません",
|
||||
"system_app_error": "システムアプリの追加に失敗しました"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "セキュリティパッチ",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "無効な boot 形式です",
|
||||
"invalid_system": "無効な system 形式です",
|
||||
"invalid_vendor": "無効な vendor 形式です"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "システムアプリを追加",
|
||||
"add": "追加",
|
||||
"current_list": "現在のシステムアプリリスト"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "アンインストールの確認",
|
||||
"uninstall_message": "Tricky Addonをアンインストールしてもよろしいですか",
|
||||
"uninstall_cancel": "キャンセル",
|
||||
"uninstall_confirm": "確認"
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "Polski",
|
||||
"system_default": "Domyślne systemowe",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Dostępne tylko w Magisk, wybrane aplikacje, które są na DenyList. Zalecane.",
|
||||
"deselect_unnecessary": "Odznacz niepotrzebne",
|
||||
"deselect_unnecessary_description": "Kategoria niepotrzebna: moduł Xposed, menedżer root, aplikacje związane z rootem i ogólne aplikacje, które nigdy nie sprawdzają stanu bootloadera. Ta opcja wymaga połączenia internetowego.",
|
||||
"add_system_app": "Dodaj aplikację systemową",
|
||||
"add_system_app_description": "Dodaj konkretną aplikację systemową do listy aplikacji.",
|
||||
"set_keybox": "Ustaw AOSP i prawidłowy klucz",
|
||||
"set_keybox_description": "Zastąp tricky store keybox.xml. Klucz AOSP zostanie zastąpiony, jeśli nie będzie już prawidłowego klucza. Opcja pobrania prawidłowego klucza wymaga połączenia internetowego.",
|
||||
"set_custom_keybox": "Ustaw niestandardowy klucz",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Odznacz wszystko",
|
||||
"select_denylist": "Wybierz z listy odrzuconych",
|
||||
"deselect_unnecessary": "Odznacz niepotrzebne",
|
||||
"add_system_app": "Dodaj aplikację systemową",
|
||||
"set_aosp_keybox": "Ustaw klucz AOSP",
|
||||
"set_valid_keybox": "Ustaw ważny klucz",
|
||||
"set_custom_keybox": "Ustaw niestandardowy klucz ",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "Nie udało się ponownie uruchomić, proszę ponownie uruchomić ręcznie",
|
||||
"custom_key_set": "Niestandardowe klucz został pomyślnie ustawiony",
|
||||
"custom_key_set_error": "Nie udało się ustawić niestandardowego klucza",
|
||||
"no_file_selected": "Nie wybrano pliku"
|
||||
"no_file_selected": "Nie wybrano pliku",
|
||||
"system_app_not_found": "Aplikacja systemowa nie znaleziona",
|
||||
"system_app_error": "Nie udało się dodać aplikacji systemowej"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Poprawka bezpieczeństwa",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Nieprawidłowy format rozruchu",
|
||||
"invalid_system": "Nieprawidłowy format systemu",
|
||||
"invalid_vendor": "Nieprawidłowy format dostawcy"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Dodaj aplikację systemową",
|
||||
"add": "Dodaj",
|
||||
"current_list": "Aktualna lista aplikacji systemowych"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Potwierdzić odinstalowanie?",
|
||||
"uninstall_message": "Czy na pewno chcesz odinstalować Tricky Addon",
|
||||
"uninstall_cancel": "Anuluj",
|
||||
"uninstall_confirm": "Potwierdź"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "Русский",
|
||||
"system_default": "Системный по умолчанию",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Доступно только в Magisk, выберите приложения, которые находятся в DenyList. Рекомендуется.",
|
||||
"deselect_unnecessary": "Отменить выбор ненужных",
|
||||
"deselect_unnecessary_description": "Ненужные категории: модули Xposed, менеджеры root, приложения, связанные с root, и общие приложения, которые никогда не проверяют статус загрузчика. Этот параметр требует подключения к интернету.",
|
||||
"add_system_app": "Добавить системное приложение",
|
||||
"add_system_app_description": "Добавить конкретное системное приложение в список приложений.",
|
||||
"set_keybox": "Установить AOSP и действующий Keybox",
|
||||
"set_keybox_description": "Замените tricky store keybox.xml. AOSP keybox будет заменен, если не будет найден действующий keybox. Опция с действующим keybox требует подключения к интернету.",
|
||||
"set_custom_keybox": "Установить пользовательский Keybox",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Отменить выбор всех",
|
||||
"select_denylist": "Выбрать из DenyList",
|
||||
"deselect_unnecessary": "Отменить выбор ненужных",
|
||||
"add_system_app": "Добавить системное приложение",
|
||||
"set_aosp_keybox": "Установить AOSP Keybox",
|
||||
"set_valid_keybox": "Установить действующий Keybox",
|
||||
"set_custom_keybox": "Установить пользовательский Keybox",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "Не удалось перезагрузить, перезагрузите вручную",
|
||||
"custom_key_set": "Пользовательский keybox успешно установлен",
|
||||
"custom_key_set_error": "Не удалось установить пользовательский keybox",
|
||||
"no_file_selected": "Файл не выбран"
|
||||
"no_file_selected": "Файл не выбран",
|
||||
"system_app_not_found": "Системное приложение не найдено",
|
||||
"system_app_error": "Не удалось добавить системное приложение"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Патч безопасности",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Неверный формат boot",
|
||||
"invalid_system": "Неверный формат system",
|
||||
"invalid_vendor": "Неверный формат vendor"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Добавить системное приложение",
|
||||
"add": "Добавить",
|
||||
"current_list": "Текущий список системных приложений"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Подтвердить удаление?",
|
||||
"uninstall_message": "Вы уверены, что хотите удалить Tricky Addon",
|
||||
"uninstall_cancel": "Отмена",
|
||||
"uninstall_confirm": "Подтвердить"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "Tagalog",
|
||||
"language": "Filipino",
|
||||
"system_default": "Default ng Sistema",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Available lang sa Magisk, piliin ang mga app na nasa DenyList. Inirerekomenda.",
|
||||
"deselect_unnecessary": "Huwag Pumili ng Hindi Kinakailangan",
|
||||
"deselect_unnecessary_description": "Hindi kinakailangang kategorya: Xposed module, root manager, root-related apps, at mga karaniwang apps na hindi kailanman nire-refresh ang bootloader status. Nangangailangan ng koneksyon sa internet.",
|
||||
"add_system_app": "Magdagdag ng System App",
|
||||
"add_system_app_description": "Magdagdag ng tiyak na system app sa listahan ng apps.",
|
||||
"set_keybox": "I-set ang AOSP at Valid Keybox",
|
||||
"set_keybox_description": "Palitan ang tricky store keybox. Palitan ang AOSP keybox kung walang valid keybox. Nangangailangan ng koneksyon sa internet ang valid keybox option.",
|
||||
"set_custom_keybox": "I-set ang Custom Keybox",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Huwag Pumili ng Lahat",
|
||||
"select_denylist": "Piliin mula sa DenyList",
|
||||
"deselect_unnecessary": "Huwag Pumili ng Hindi Kinakailangan",
|
||||
"add_system_app": "Magdagdag ng System App",
|
||||
"set_aosp_keybox": "I-set ang AOSP Keybox",
|
||||
"set_valid_keybox": "I-set ang Valid Keybox",
|
||||
"set_custom_keybox": "I-set ang Custom Keybox",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "Nabigo ang pag-reboot, pakisubukang mag-reboot nang manu-mano",
|
||||
"custom_key_set": "Matagumpay na na-set ang Custom Keybox",
|
||||
"custom_key_set_error": "Nabigong i-set ang Custom Keybox",
|
||||
"no_file_selected": "Walang napiling file"
|
||||
"no_file_selected": "Walang napiling file",
|
||||
"system_app_not_found": "Walang natagpuan na system app",
|
||||
"system_app_error": "Nabigong dagdag ang system app"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Security Patch",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Invalid boot format",
|
||||
"invalid_system": "Invalid system format",
|
||||
"invalid_vendor": "Invalid vendor format"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Magdagdag ng System App",
|
||||
"add": "Idagdag",
|
||||
"current_list": "Kasalukuyang Listahan ng System App"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Kumpirmahin ang Pag-uninstall?",
|
||||
"uninstall_message": "Sigurado ka bang gusto mong i-uninstall ang Tricky Addon",
|
||||
"uninstall_cancel": "Kanselahin",
|
||||
"uninstall_confirm": "Kumpirmahin"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "Türkçe",
|
||||
"system_default": "Sistem Varsayılanı",
|
||||
"header": {
|
||||
"title": "Tricky Addon"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "Yalnızca Magisk'te mevcut, Reddetme Listesindeki uygulamaları seç. Tavsiye edilir.",
|
||||
"deselect_unnecessary": "Gereksizleri Seçme",
|
||||
"deselect_unnecessary_description": "Gereksiz kategori: Xposed modülü, root yöneticisi, root ile ilgili uygulamalar ve asla bootloader durumunu kontrol etmeyen genel uygulamalar. Bu seçenek internet bağlantısı gerektirir.",
|
||||
"add_system_app": "Sistem Uygulaması Ekle",
|
||||
"add_system_app_description": "Belirli bir sistem uygulamasını uygulama listesine ekleyin.",
|
||||
"set_keybox": "AOSP & Geçerli Keybox Ayarla",
|
||||
"set_keybox_description": "Tricky Store'daki keybox.xml dosyasını değiştirir. Eğer geçerli bir keybox yoksa AOSP keybox ile değiştirilecektir. Geçerli keybox seçeneği internet bağlantısı gerektirir.",
|
||||
"set_custom_keybox": "Özel Keybox Ayarla",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "Tüm Seçimleri Kaldır",
|
||||
"select_denylist": "Reddetme Listesinden Seç",
|
||||
"deselect_unnecessary": "Gereksizleri Seçme",
|
||||
"add_system_app": "Sistem Uygulaması Ekle",
|
||||
"set_aosp_keybox": "AOSP Keybox Ayarla",
|
||||
"set_valid_keybox": "Geçerli Keybox Ayarla",
|
||||
"set_custom_keybox": "Özel Keybox Ayarla",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "Yeniden başlatma başarısız, lütfen manuel olarak yeniden başlatın",
|
||||
"custom_key_set": "Özel keybox başarıyla ayarlandı",
|
||||
"custom_key_set_error": "Özel keybox ayarlanamadı",
|
||||
"no_file_selected": "Dosya seçilmedi"
|
||||
"no_file_selected": "Dosya seçilmedi",
|
||||
"system_app_not_found": "Sistem uygulaması bulunamadı",
|
||||
"system_app_error": "Sistem uygulaması ekleme hatası"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "Güvenlik Yaması",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "Geçersiz boot formatı",
|
||||
"invalid_system": "Geçersiz system formatı",
|
||||
"invalid_vendor": "Geçersiz vendor formatı"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "Sistem Uygulaması Ekle",
|
||||
"add": "Ekle",
|
||||
"current_list": "Mevcut Sistem Uygulamaları Listesi"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "Kaldırma İşlemi Onaylansın mı?",
|
||||
"uninstall_message": "Tricky Addon'u kaldırmak istediğinizden emin misiniz",
|
||||
"uninstall_cancel": "İptal",
|
||||
"uninstall_confirm": "Onayla"
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "简体中文",
|
||||
"system_default": "系统默认",
|
||||
"header": {
|
||||
"title": "TS 插件"
|
||||
},
|
||||
@@ -13,8 +14,10 @@
|
||||
"select_description": "选择或取消选择当前界面中的所有应用。",
|
||||
"select_denylist": "从排除列表中选择",
|
||||
"select_denylist_description": "仅适用于 Magisk,选择在排除列表中的应用。推荐使用。",
|
||||
"deselect_unnecessary": "取消选择非必应用",
|
||||
"deselect_unnecessary": "取消选择非必要应用",
|
||||
"deselect_unnecessary_description": "非必要分类:Xposed 模块、root 管理器、与 root 相关的应用,以及从不检查 bootloader 状态的通用应用。此功能需连网使用。",
|
||||
"add_system_app": "添加系统应用",
|
||||
"add_system_app_description": "添加特定系统应用到应用列表。",
|
||||
"set_keybox": "设置 AOSP & 有效密钥",
|
||||
"set_keybox_description": "替换 Tricky Store 的密钥(keybox.xml)。如果没有有效密钥,将替换为 AOSP 密钥。有效密钥选项需连网使用。",
|
||||
"set_custom_keybox": "设置自定义密钥",
|
||||
@@ -46,7 +49,8 @@
|
||||
"select_all": "全选",
|
||||
"deselect_all": "取消全选",
|
||||
"select_denylist": "从排除列表中选择",
|
||||
"deselect_unnecessary": "取消选择非必应用",
|
||||
"deselect_unnecessary": "取消选择非必要应用",
|
||||
"add_system_app": "添加系统应用",
|
||||
"set_aosp_keybox": "设置 AOSP 密钥",
|
||||
"set_valid_keybox": "设置有效密钥",
|
||||
"set_custom_keybox": "设置自定义密钥",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "重启失败,请手动重启",
|
||||
"custom_key_set": "成功设置自定义密钥",
|
||||
"custom_key_set_error": "设置自定义密钥失败",
|
||||
"no_file_selected": "未选择文件"
|
||||
"no_file_selected": "未选择文件",
|
||||
"system_app_not_found": "未找到该系统应用",
|
||||
"system_app_error": "系统应用添加失败"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "安全补丁",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "无效 boot 格式",
|
||||
"invalid_system": "无效 system 格式",
|
||||
"invalid_vendor": "无效 vendor 格式"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "添加系统应用",
|
||||
"add": "添加",
|
||||
"current_list": "当前系统应用列表"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "确认卸载?",
|
||||
"uninstall_message": "您确定要卸载 TS 插件吗",
|
||||
"uninstall_cancel": "取消",
|
||||
"uninstall_confirm": "确认"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
{
|
||||
"language": "繁體中文",
|
||||
"system_default": "系統預設",
|
||||
"header": {
|
||||
"title": "TS 插件"
|
||||
},
|
||||
@@ -15,6 +16,8 @@
|
||||
"select_denylist_description": "僅適用於 Magisk,選擇在排除列表中的應用。推薦使用。",
|
||||
"deselect_unnecessary": "取消選擇非必要應用",
|
||||
"deselect_unnecessary_description": "非必要分類:Xposed 模組、root 管理器、與 root 相關的應用,以及從不檢查 bootloader 狀態的通用應用。此功能需連網使用。",
|
||||
"add_system_app": "添加系統應用",
|
||||
"add_system_app_description": "添加特定系統應用到應用列表。",
|
||||
"set_keybox": "設置 AOSP & 有效密鑰",
|
||||
"set_keybox_description": "替換 Tricky Store 的密鑰(keybox.xml)。如果沒有有效密鑰,將替換為 AOSP 密鑰。有效密鑰選項需連網使用。",
|
||||
"set_custom_keybox": "設置自定義密鑰",
|
||||
@@ -47,6 +50,7 @@
|
||||
"deselect_all": "取消全選",
|
||||
"select_denylist": "從排除列表中選擇",
|
||||
"deselect_unnecessary": "取消選擇非必要應用",
|
||||
"add_system_app": "添加系統應用",
|
||||
"set_aosp_keybox": "設置 AOSP 密鑰",
|
||||
"set_valid_keybox": "設置有效密鑰",
|
||||
"set_custom_keybox": "設置自定義密鑰",
|
||||
@@ -91,7 +95,9 @@
|
||||
"reboot_fail": "重啟失敗,請手動重啟",
|
||||
"custom_key_set": "成功設置自定義密鑰",
|
||||
"custom_key_set_error": "設置自定義密鑰失敗",
|
||||
"no_file_selected": "未選擇文件"
|
||||
"no_file_selected": "未選擇文件",
|
||||
"system_app_not_found": "未找到該系統應用",
|
||||
"system_app_error": "系統應用添加失敗"
|
||||
},
|
||||
"security_patch": {
|
||||
"title": "安全補丁",
|
||||
@@ -111,5 +117,16 @@
|
||||
"invalid_boot": "無效 boot 格式",
|
||||
"invalid_system": "無效 system 格式",
|
||||
"invalid_vendor": "無效 vendor 格式"
|
||||
},
|
||||
"add_system_app": {
|
||||
"title": "添加系統應用",
|
||||
"add": "添加",
|
||||
"current_list": "當前系統應用列表"
|
||||
},
|
||||
"confirmation": {
|
||||
"uninstall_title": "確認卸載?",
|
||||
"uninstall_message": "您確定要卸載 TS 插件嗎",
|
||||
"uninstall_cancel": "取消",
|
||||
"uninstall_confirm": "確認"
|
||||
}
|
||||
}
|
||||
@@ -1,46 +1,38 @@
|
||||
import { execCommand, linkRedirect } from './main.js';
|
||||
|
||||
const telegramLink = document.getElementById('telegram');
|
||||
const githubLink = document.getElementById('github');
|
||||
import { linkRedirect } from './main.js';
|
||||
|
||||
// Function to show about overlay
|
||||
document.getElementById("about").addEventListener("click", () => {
|
||||
const aboutOverlay = document.getElementById('about-overlay');
|
||||
const aboutMenu = document.getElementById('about-menu');
|
||||
const aboutContent = document.querySelector('.about-menu');
|
||||
const closeAbout = document.getElementById('close-about');
|
||||
const showMenu = () => {
|
||||
aboutOverlay.style.display = 'flex';
|
||||
setTimeout(() => {
|
||||
aboutOverlay.style.opacity = '1';
|
||||
aboutMenu.style.opacity = '1';
|
||||
}, 10);
|
||||
document.body.style.overflow = 'hidden';
|
||||
};
|
||||
|
||||
// Show about menu
|
||||
document.body.classList.add("no-scroll");
|
||||
aboutOverlay.style.display = 'flex';
|
||||
setTimeout(() => {
|
||||
aboutOverlay.style.opacity = '1';
|
||||
aboutContent.classList.add('open');
|
||||
}, 10);
|
||||
|
||||
const hideMenu = () => {
|
||||
document.body.classList.remove("no-scroll");
|
||||
aboutOverlay.style.opacity = '0';
|
||||
aboutMenu.style.opacity = '0';
|
||||
aboutContent.classList.remove('open');
|
||||
setTimeout(() => {
|
||||
aboutOverlay.style.display = 'none';
|
||||
document.body.style.overflow = 'auto';
|
||||
}, 200);
|
||||
};
|
||||
showMenu();
|
||||
closeAbout.addEventListener('click', (event) => {
|
||||
event.stopPropagation();
|
||||
hideMenu();
|
||||
});
|
||||
|
||||
closeAbout.addEventListener("click", hideMenu);
|
||||
aboutOverlay.addEventListener('click', (event) => {
|
||||
if (!aboutMenu.contains(event.target)) {
|
||||
hideMenu();
|
||||
}
|
||||
if (event.target === aboutOverlay) hideMenu();
|
||||
});
|
||||
menu.addEventListener('click', (event) => event.stopPropagation());
|
||||
});
|
||||
|
||||
// Event listener for link redirect
|
||||
telegramLink.addEventListener('click', function() {
|
||||
document.getElementById('telegram').addEventListener('click', function() {
|
||||
linkRedirect('https://t.me/kowchannel');
|
||||
});
|
||||
githubLink.addEventListener('click', function() {
|
||||
document.getElementById('github').addEventListener('click', function() {
|
||||
linkRedirect('https://github.com/KOWX712/Tricky-Addon-Update-Target-List');
|
||||
});
|
||||
@@ -1,69 +1,62 @@
|
||||
import { basePath, execCommand, floatingBtn, appsWithExclamation, appsWithQuestion, toast } from './main.js';
|
||||
import { basePath, execCommand, hideFloatingBtn, appsWithExclamation, appsWithQuestion, toast } from './main.js';
|
||||
|
||||
const appTemplate = document.getElementById('app-template').content;
|
||||
const modeOverlay = document.querySelector('.mode-overlay');
|
||||
export const appListContainer = document.getElementById('apps-list');
|
||||
export const updateCard = document.getElementById('update-card');
|
||||
export let modeActive = false;
|
||||
|
||||
// Fetch and render applist
|
||||
export async function fetchAppList() {
|
||||
try {
|
||||
// fetch target list
|
||||
let targetList = [];
|
||||
try {
|
||||
const targetFileContent = await execCommand('cat /data/adb/tricky_store/target.txt');
|
||||
targetList = processTargetList(targetFileContent);
|
||||
console.log("Current target list:", targetList);
|
||||
} catch (error) {
|
||||
toast("Failed to read target.txt!");
|
||||
console.error("Failed to read target.txt file:", error);
|
||||
}
|
||||
|
||||
let applistMap = {};
|
||||
try {
|
||||
const applistResult = await execCommand(`cat ${basePath}common/tmp/applist`);
|
||||
applistMap = applistResult
|
||||
.split("\n")
|
||||
.reduce((map, line) => {
|
||||
const match = line.match(/app-name:\s*(.+),\s*package-name:\s*(.+)/);
|
||||
if (match) {
|
||||
const appName = match[1].trim();
|
||||
const packageName = match[2].trim();
|
||||
map[packageName] = appName;
|
||||
}
|
||||
return map;
|
||||
}, {});
|
||||
console.log("Applist loaded successfully.");
|
||||
} catch (error) {
|
||||
console.warn("Applist file not found or could not be loaded. Skipping applist lookup.");
|
||||
}
|
||||
// fetch applist
|
||||
const response = await fetch('applist.json');
|
||||
const appList = await response.json();
|
||||
const appNameMap = appList.reduce((map, app) => {
|
||||
map[app.package_name] = app.app_name;
|
||||
return map;
|
||||
}, {});
|
||||
|
||||
const result = await execCommand(`
|
||||
pm list packages -3 | awk -F: '{print $2}'
|
||||
pm list packages -s | awk -F: '{print $2}' | grep -Ex "com.google.android.gms|com.google.android.gsf|com.android.vending" 2>/dev/null || true
|
||||
`);
|
||||
const appEntries = result
|
||||
.split("\n")
|
||||
.map(line => {
|
||||
const packageName = line.trim();
|
||||
const appName = applistMap[packageName] || null;
|
||||
return { appName, packageName };
|
||||
})
|
||||
.filter(entry => entry.packageName);
|
||||
for (const entry of appEntries) {
|
||||
if (!entry.appName) {
|
||||
try {
|
||||
const apkPath = await execCommand(`pm path ${entry.packageName} | grep "base.apk" | awk -F: '{print $2}' | tr -d '\\r'`);
|
||||
if (apkPath) {
|
||||
const appName = await execCommand(`${basePath}common/aapt dump badging ${apkPath.trim()} 2>/dev/null | grep "application-label:" | sed "s/application-label://; s/'//g"`);
|
||||
entry.appName = appName.trim() || "Unknown App";
|
||||
} else {
|
||||
entry.appName = "Unknown App";
|
||||
}
|
||||
} catch (error) {
|
||||
entry.appName = "Unknown App";
|
||||
// Get installed packages first
|
||||
let appEntries = [], installedPackages = [];
|
||||
try {
|
||||
installedPackages = await execCommand(`
|
||||
pm list packages -3 | awk -F: '{print $2}'
|
||||
[ -s "/data/adb/tricky_store/system_app" ] && SYSTEM_APP=$(cat "/data/adb/tricky_store/system_app" | tr '\n' '|' | sed 's/|*$//') || SYSTEM_APP=""
|
||||
[ -z "$SYSTEM_APP" ] || pm list packages -s | awk -F: '{print $2}' | grep -Ex "$SYSTEM_APP" 2>/dev/null || true
|
||||
`)
|
||||
installedPackages = installedPackages.split("\n").map(line => line.trim()).filter(Boolean);
|
||||
appEntries = await Promise.all(installedPackages.map(async packageName => {
|
||||
if (appNameMap[packageName]) {
|
||||
return {
|
||||
appName: appNameMap[packageName],
|
||||
packageName
|
||||
};
|
||||
}
|
||||
}
|
||||
const appName = await execCommand(`
|
||||
base_apk=$(pm path ${packageName} | grep "base.apk" | awk -F: '{print $2}')
|
||||
[ -n "$base_apk" ] || base_apk=$(pm path ${packageName} | grep ".apk" | awk -F: '{print $2}')
|
||||
${basePath}/common/aapt dump badging $base_apk 2>/dev/null | grep "application-label:" | sed "s/application-label://; s/'//g"
|
||||
`);
|
||||
return {
|
||||
appName: appName.trim() || packageName,
|
||||
packageName
|
||||
};
|
||||
}));
|
||||
} catch (error) {
|
||||
appEntries = appList.map(app => ({
|
||||
appName: app.app_name,
|
||||
packageName: app.package_name
|
||||
}));
|
||||
}
|
||||
|
||||
// Sort
|
||||
@@ -103,17 +96,21 @@ export async function fetchAppList() {
|
||||
}
|
||||
|
||||
const nameElement = appElement.querySelector(".name");
|
||||
nameElement.innerHTML = `<strong>${appName || "Unknown App"}</strong><br>${packageName}`;
|
||||
nameElement.innerHTML = `
|
||||
<div class="app-info">
|
||||
<div class="app-name"><strong>${appName}</strong></div>
|
||||
<div class="package-name">${packageName}</div>
|
||||
</div>
|
||||
`;
|
||||
const checkbox = appElement.querySelector(".checkbox");
|
||||
checkbox.checked = targetList.includes(packageName);
|
||||
appListContainer.appendChild(appElement);
|
||||
});
|
||||
console.log("App list with names and packages rendered successfully.");
|
||||
} catch (error) {
|
||||
toast("Failed to fetch app list!");
|
||||
console.error("Failed to fetch or render app list with names:", error);
|
||||
}
|
||||
floatingBtn.style.transform = 'translateY(0)';
|
||||
hideFloatingBtn(false);
|
||||
toggleableCheckbox();
|
||||
if (appListContainer.firstChild !== updateCard) {
|
||||
appListContainer.insertBefore(updateCard, appListContainer.firstChild);
|
||||
@@ -201,7 +198,6 @@ function setupModeMenu() {
|
||||
function showMode(card) {
|
||||
const modeElement = card.querySelector(".mode");
|
||||
if (modeElement) {
|
||||
modeActive = true;
|
||||
modeElement.style.display = "flex";
|
||||
modeOverlay.style.display = "flex";
|
||||
setTimeout(() => {
|
||||
@@ -212,7 +208,6 @@ function setupModeMenu() {
|
||||
function hideAllModes() {
|
||||
const allModeElements = appListContainer.querySelectorAll(".mode");
|
||||
allModeElements.forEach((modeElement) => {
|
||||
modeActive = false;
|
||||
modeElement.classList.remove('show');
|
||||
modeOverlay.style.display = "none";
|
||||
setTimeout(() => {
|
||||
|
||||
@@ -1,36 +1,33 @@
|
||||
import { execCommand, showPrompt } from './main.js';
|
||||
|
||||
const bootHashOverlay = document.getElementById('boot-hash-overlay');
|
||||
const card = document.getElementById('boot-hash-card');
|
||||
const bootHash = document.querySelector('.boot-hash-card');
|
||||
const inputBox = document.getElementById('boot-hash-input');
|
||||
const saveButton = document.getElementById('boot-hash-save-button');
|
||||
|
||||
// Remove empty spaces from input
|
||||
// Remove empty spaces from input and convert to lowercase
|
||||
window.trimInput = (input) => {
|
||||
input.value = input.value.replace(/\s+/g, '');
|
||||
input.value = input.value.replace(/\s+/g, '').toLowerCase();
|
||||
};
|
||||
|
||||
// Function to handle Verified Boot Hash
|
||||
document.getElementById("boot-hash").addEventListener("click", async () => {
|
||||
const showCard = () => {
|
||||
bootHashOverlay.style.display = "flex";
|
||||
card.style.display = "flex";
|
||||
requestAnimationFrame(() => {
|
||||
bootHashOverlay.classList.add("show");
|
||||
card.classList.add("show");
|
||||
});
|
||||
document.body.style.overflow = "hidden";
|
||||
};
|
||||
const closeCard = () => {
|
||||
bootHashOverlay.classList.remove("show");
|
||||
card.classList.remove("show");
|
||||
// Display boot hash menu
|
||||
document.body.classList.add("no-scroll");
|
||||
bootHashOverlay.style.display = "flex";
|
||||
setTimeout(() => {
|
||||
bootHashOverlay.style.opacity = 1;
|
||||
bootHash.classList.add('open');
|
||||
}, 10);
|
||||
|
||||
const closeBootHashMenu = () => {
|
||||
document.body.classList.remove("no-scroll");
|
||||
bootHashOverlay.style.opacity = 0;
|
||||
bootHash.classList.remove('open');
|
||||
setTimeout(() => {
|
||||
bootHashOverlay.style.display = "none";
|
||||
card.style.display = "none";
|
||||
document.body.style.overflow = "auto";
|
||||
}, 200);
|
||||
};
|
||||
showCard();
|
||||
try {
|
||||
const bootHashContent = await execCommand("cat /data/adb/boot_hash");
|
||||
const validHash = bootHashContent
|
||||
@@ -44,17 +41,23 @@ document.getElementById("boot-hash").addEventListener("click", async () => {
|
||||
saveButton.addEventListener("click", async () => {
|
||||
const inputValue = inputBox.value.trim();
|
||||
try {
|
||||
await execCommand(`echo "${inputValue}" > /data/adb/boot_hash`);
|
||||
await execCommand(`su -c resetprop -n ro.boot.vbmeta.digest ${inputValue}`);
|
||||
await execCommand(`
|
||||
PATH=/data/adb/ap/bin:/data/adb/ksu/bin:/data/adb/magisk:$PATH
|
||||
resetprop -n ro.boot.vbmeta.digest ${inputValue}
|
||||
[ -z "${inputValue}" ] && rm -f /data/adb/boot_hash || {
|
||||
echo "${inputValue}" > /data/adb/boot_hash
|
||||
chmod 644 /data/adb/boot_hash
|
||||
}
|
||||
`);
|
||||
showPrompt("prompt.boot_hash_set");
|
||||
closeCard();
|
||||
closeBootHashMenu();
|
||||
} catch (error) {
|
||||
console.error("Failed to update boot_hash:", error);
|
||||
showPrompt("prompt.boot_hash_set_error", false);
|
||||
}
|
||||
});
|
||||
bootHashOverlay.addEventListener("click", (event) => {
|
||||
if (event.target === bootHashOverlay) closeCard();
|
||||
if (event.target === bootHashOverlay) closeBootHashMenu();
|
||||
});
|
||||
|
||||
// Enter to save
|
||||
|
||||
@@ -1,21 +1,21 @@
|
||||
const helpButton = document.getElementById('help-button');
|
||||
const helpOverlay = document.getElementById('help-overlay');
|
||||
const helpContent = document.querySelector('.help-menu');
|
||||
const closeHelp = document.getElementById('close-help');
|
||||
const helpList = document.getElementById('help-list');
|
||||
|
||||
// Open help menu
|
||||
helpButton.addEventListener("click", () => {
|
||||
helpOverlay.classList.remove("hide");
|
||||
helpOverlay.style.display = "flex";
|
||||
requestAnimationFrame(() => {
|
||||
helpOverlay.classList.add("show");
|
||||
});
|
||||
document.body.classList.add("no-scroll");
|
||||
helpOverlay.style.display = "flex";
|
||||
setTimeout(() => {
|
||||
helpOverlay.style.opacity = 1;
|
||||
helpContent.classList.add('open');
|
||||
}, 10);
|
||||
});
|
||||
|
||||
const hideHelpOverlay = () => {
|
||||
helpOverlay.classList.remove("show");
|
||||
helpOverlay.classList.add("hide");
|
||||
helpOverlay.style.opacity = 0;
|
||||
helpContent.classList.remove('open');
|
||||
document.body.classList.remove("no-scroll");
|
||||
setTimeout(() => {
|
||||
helpOverlay.style.display = "none";
|
||||
|
||||
@@ -1,57 +1,60 @@
|
||||
import { basePath, execCommand, toast } from './main.js';
|
||||
|
||||
const languageButton = document.querySelector('.language-button');
|
||||
const languageMenu = document.querySelector('.language-menu');
|
||||
const languageOptions = document.querySelectorAll('.language-option');
|
||||
const languageOverlay = document.getElementById('language-overlay');
|
||||
|
||||
export let translations = {};
|
||||
let currentLang = 'en-US';
|
||||
let availableLanguages = ['en-US'];
|
||||
|
||||
// Function to check for available language
|
||||
export async function initializeAvailableLanguages() {
|
||||
try {
|
||||
const multiLang = await execCommand(`find ${basePath}webui/locales -type f -name "*.json" ! -name "A-template.json" -exec basename -s .json {} \\;`);
|
||||
availableLanguages = multiLang.trim().split('\n');
|
||||
generateLanguageMenu();
|
||||
} catch (error) {
|
||||
toast("Failed to get available langauge!");
|
||||
console.error('Failed to fetch available languages:', error);
|
||||
availableLanguages = ['en-US'];
|
||||
}
|
||||
}
|
||||
|
||||
// Function to detect user's default language
|
||||
export function detectUserLanguage() {
|
||||
/**
|
||||
* Detect user's default language
|
||||
* @returns {Promise<string>} - Detected language code
|
||||
*/
|
||||
async function detectUserLanguage() {
|
||||
const userLang = navigator.language || navigator.userLanguage;
|
||||
const langCode = userLang.split('-')[0];
|
||||
if (availableLanguages.includes(userLang)) {
|
||||
return userLang;
|
||||
} else if (availableLanguages.includes(langCode)) {
|
||||
return langCode;
|
||||
} else {
|
||||
|
||||
try {
|
||||
// Fetch available languages
|
||||
const availableResponse = await fetch('locales/available-lang.json');
|
||||
const availableData = await availableResponse.json();
|
||||
availableLanguages = availableData.languages;
|
||||
await generateLanguageMenu();
|
||||
|
||||
// Fetch preferred language
|
||||
const prefered_language_code = localStorage.getItem('trickyAddonLanguage');
|
||||
|
||||
// Check if preferred language is valid
|
||||
if (prefered_language_code !== 'default' && availableLanguages.includes(prefered_language_code)) {
|
||||
return prefered_language_code;
|
||||
} else if (availableLanguages.includes(userLang)) {
|
||||
return userLang;
|
||||
} else if (availableLanguages.includes(langCode)) {
|
||||
return langCode;
|
||||
} else {
|
||||
return 'en-US';
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error detecting user language:', error);
|
||||
return 'en-US';
|
||||
}
|
||||
}
|
||||
|
||||
// Load translations dynamically based on the selected language
|
||||
export async function loadTranslations(lang) {
|
||||
try {
|
||||
const response = await fetch(`/locales/${lang}.json`);
|
||||
translations = await response.json();
|
||||
applyTranslations();
|
||||
} catch (error) {
|
||||
toast(`Failed to load translation for ${lang}!`);
|
||||
console.error(`Error loading translations for ${lang}:`, error);
|
||||
if (lang !== 'en-US') {
|
||||
console.log("Falling back to English.");
|
||||
loadTranslations('en-US');
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Load translations dynamically based on the selected language
|
||||
* @returns {Promise<void>}
|
||||
*/
|
||||
export async function loadTranslations() {
|
||||
const lang = await detectUserLanguage();
|
||||
const response = await fetch(`locales/${lang}.json`);
|
||||
translations = await response.json();
|
||||
applyTranslations();
|
||||
}
|
||||
|
||||
// Function to apply translations to all elements with data-i18n attributes
|
||||
/**
|
||||
* Apply translations to all elements with data-i18n attributes
|
||||
* @returns {void}
|
||||
*/
|
||||
function applyTranslations() {
|
||||
document.querySelectorAll("[data-i18n]").forEach((el) => {
|
||||
const keyString = el.getAttribute("data-i18n");
|
||||
@@ -66,7 +69,10 @@ function applyTranslations() {
|
||||
});
|
||||
}
|
||||
|
||||
// Function to setup the language menu
|
||||
/**
|
||||
* Function to setup the language menu
|
||||
* @returns {void}
|
||||
*/
|
||||
export function setupLanguageMenu() {
|
||||
languageButton.addEventListener("click", (event) => {
|
||||
event.stopPropagation();
|
||||
@@ -105,18 +111,31 @@ export function setupLanguageMenu() {
|
||||
languageMenu.addEventListener("click", (e) => {
|
||||
if (e.target.classList.contains("language-option")) {
|
||||
const lang = e.target.getAttribute("data-lang");
|
||||
localStorage.setItem('trickyAddonLanguage', lang);
|
||||
loadTranslations(lang);
|
||||
closeLanguageMenu();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Function to generate the language menu dynamically
|
||||
/**
|
||||
* Generate the language menu dynamically
|
||||
* Refer available-lang.json in ./locales for list of languages
|
||||
* @returns {Promise<void>}
|
||||
*/
|
||||
async function generateLanguageMenu() {
|
||||
languageMenu.innerHTML = '';
|
||||
|
||||
// Add System Default option
|
||||
const defaultButton = document.createElement('button');
|
||||
defaultButton.classList.add('language-option', 'ripple-element');
|
||||
defaultButton.setAttribute('data-lang', 'default');
|
||||
defaultButton.setAttribute('data-i18n', 'system_default');
|
||||
languageMenu.appendChild(defaultButton);
|
||||
|
||||
const languagePromises = availableLanguages.map(async (lang) => {
|
||||
try {
|
||||
const response = await fetch(`/locales/${lang}.json`);
|
||||
const response = await fetch(`locales/${lang}.json`);
|
||||
const data = await response.json();
|
||||
return { lang, name: data.language || lang };
|
||||
} catch (error) {
|
||||
|
||||
@@ -1,12 +1,11 @@
|
||||
import { appListContainer, fetchAppList, modeActive } from './applist.js';
|
||||
import { initializeAvailableLanguages, detectUserLanguage, loadTranslations, setupLanguageMenu, translations } from './language.js';
|
||||
import { aospkb } from './menu_option.js';
|
||||
import { appListContainer, fetchAppList } from './applist.js';
|
||||
import { loadTranslations, setupLanguageMenu, translations } from './language.js';
|
||||
import { setupSystemAppMenu } from './menu_option.js';
|
||||
import { searchMenuContainer, searchInput, clearBtn, setupMenuToggle } from './search_menu.js';
|
||||
import { updateCheck } from './update.js';
|
||||
import { securityPatch } from './security_patch.js';
|
||||
|
||||
// Header Elements
|
||||
const headerBlock = document.querySelector('.header-block');
|
||||
const title = document.querySelector('.header');
|
||||
export const noConnection = document.querySelector('.no-connection');
|
||||
|
||||
@@ -15,32 +14,41 @@ const permissionPopup = document.getElementById('permission-popup');
|
||||
const loadingIndicator = document.querySelector('.loading');
|
||||
const prompt = document.getElementById('prompt');
|
||||
const floatingCard = document.querySelector('.floating-card');
|
||||
export const floatingBtn = document.querySelector('.floating-btn');
|
||||
const floatingBtn = document.querySelector('.floating-btn');
|
||||
|
||||
export const basePath = "set-path";
|
||||
export let basePath;
|
||||
export const appsWithExclamation = [];
|
||||
export const appsWithQuestion = [];
|
||||
const ADDITIONAL_APPS = [ "android", "com.android.vending", "com.google.android.gms", "io.github.vvb2060.keyattestation", "io.github.vvb2060.mahoshojo", "icu.nullptr.nativetest" ]; // Always keep default apps in target.txt
|
||||
const ADDITIONAL_APPS = [ "android", "com.google.android.gms", "io.github.vvb2060.keyattestation", "io.github.vvb2060.mahoshojo", "icu.nullptr.nativetest" ]; // Always keep default apps in target.txt
|
||||
|
||||
// Variables
|
||||
let e = 0;
|
||||
let isRefreshing = false;
|
||||
let MMRL_API = true;
|
||||
|
||||
// Function to set basePath
|
||||
async function getBasePath() {
|
||||
try {
|
||||
await execCommand('[ -d /data/adb/modules/.TA_utl ]');
|
||||
basePath = "/data/adb/modules/.TA_utl"
|
||||
} catch (error) {
|
||||
basePath = "/data/adb/modules/TA_utl"
|
||||
}
|
||||
}
|
||||
|
||||
// Function to load the version from module.prop
|
||||
async function getModuleVersion() {
|
||||
const moduleVersion = document.getElementById('module-version');
|
||||
try {
|
||||
const version = await execCommand(`grep '^version=' ${basePath}common/update/module.prop | cut -d'=' -f2`);
|
||||
const version = await execCommand(`grep '^version=' ${basePath}/common/update/module.prop | cut -d'=' -f2`);
|
||||
moduleVersion.textContent = version;
|
||||
} catch (error) {
|
||||
console.error("Failed to read version from module.prop:", error);
|
||||
updateVersion("Error");
|
||||
}
|
||||
}
|
||||
|
||||
// Function to refresh app list
|
||||
async function refreshAppList() {
|
||||
export async function refreshAppList() {
|
||||
isRefreshing = true;
|
||||
title.style.transform = 'translateY(0)';
|
||||
searchMenuContainer.style.transform = 'translateY(0)';
|
||||
@@ -55,7 +63,7 @@ async function refreshAppList() {
|
||||
if (noConnection.style.display === "flex") {
|
||||
try {
|
||||
updateCheck();
|
||||
await execCommand(`[ -f ${basePath}common/tmp/exclude-list ] && rm -f "${basePath}common/tmp/exclude-list"`);
|
||||
await execCommand(`[ -f ${basePath}/common/tmp/exclude-list ] && rm -f "${basePath}/common/tmp/exclude-list"`);
|
||||
} catch (error) {
|
||||
toast("Failed!");
|
||||
console.error("Error occurred:", error);
|
||||
@@ -76,12 +84,7 @@ async function checkTrickyStoreVersion() {
|
||||
TS_version=$(grep "versionCode=" "/data/adb/modules/tricky_store/module.prop" | cut -d'=' -f2)
|
||||
[ "$TS_version" -ge 158 ] || echo "NO"
|
||||
`);
|
||||
if (version.trim() !== "NO") {
|
||||
console.log("Tricky Store version is 158 or higher, displaying element.");
|
||||
securityPatchElement.style.display = "flex";
|
||||
} else {
|
||||
console.log("Tricky Store version is below 158, leaving security patch element hidden.");
|
||||
}
|
||||
if (version.trim() !== "NO") securityPatchElement.style.display = "flex";
|
||||
} catch (error) {
|
||||
toast("Failed to check Tricky Store version!");
|
||||
console.error("Error while checking Tricky Store version:", error);
|
||||
@@ -93,12 +96,7 @@ async function checkMagisk() {
|
||||
const selectDenylistElement = document.getElementById('select-denylist');
|
||||
try {
|
||||
const magiskEnv = await execCommand(`command -v magisk >/dev/null 2>&1 || echo "NO"`);
|
||||
if (magiskEnv.trim() !== "NO") {
|
||||
console.log("Denylist conditions met, displaying element.");
|
||||
selectDenylistElement.style.display = "flex";
|
||||
} else {
|
||||
console.log("not running on Magisk, leaving denylist element hidden.");
|
||||
}
|
||||
if (magiskEnv.trim() !== "NO") selectDenylistElement.style.display = "flex";
|
||||
} catch (error) {
|
||||
toast("Failed to check Magisk!");
|
||||
console.error("Error while checking denylist conditions:", error);
|
||||
@@ -114,11 +112,7 @@ export function showPrompt(key, isSuccess = true, duration = 3000) {
|
||||
clearTimeout(window.promptTimeout);
|
||||
}
|
||||
setTimeout(() => {
|
||||
if (typeof ksu !== 'undefined' && ksu.mmrl) {
|
||||
prompt.style.transform = 'translateY(calc((var(--window-inset-bottom) + 60%) * -1))';
|
||||
} else {
|
||||
prompt.style.transform = 'translateY(-60%)';
|
||||
}
|
||||
prompt.style.transform = 'translateY(calc((var(--window-inset-bottom, 0px) + 60%) * -1))';
|
||||
window.promptTimeout = setTimeout(() => {
|
||||
prompt.style.transform = 'translateY(100%)';
|
||||
}, duration);
|
||||
@@ -155,7 +149,6 @@ document.getElementById("save").addEventListener("click", async () => {
|
||||
});
|
||||
const updatedTargetContent = modifiedAppsList.join("\n");
|
||||
await execCommand(`echo "${updatedTargetContent}" | sort -u > /data/adb/tricky_store/target.txt`);
|
||||
console.log("target.txt updated successfully.");
|
||||
showPrompt("prompt.saved_target");
|
||||
for (const app of appsWithExclamation) {
|
||||
await execCommand(`sed -i 's/^${app}$/${app}!/' /data/adb/tricky_store/target.txt`);
|
||||
@@ -163,7 +156,6 @@ document.getElementById("save").addEventListener("click", async () => {
|
||||
for (const app of appsWithQuestion) {
|
||||
await execCommand(`sed -i 's/^${app}$/${app}?/' /data/adb/tricky_store/target.txt`);
|
||||
}
|
||||
console.log("App names modified in target.txt.");
|
||||
} catch (error) {
|
||||
console.error("Failed to update target.txt:", error);
|
||||
showPrompt("prompt.save_error", false);
|
||||
@@ -172,40 +164,61 @@ document.getElementById("save").addEventListener("click", async () => {
|
||||
});
|
||||
|
||||
// Uninstall WebUI
|
||||
document.querySelector(".uninstall-container").addEventListener("click", async () => {
|
||||
document.querySelector(".uninstall-container").addEventListener("click", () => {
|
||||
const uninstallOverlay = document.getElementById("uninstall-confirmation-overlay");
|
||||
const uninstallContent = document.querySelector('.uninstall-confirmation');
|
||||
const cancelButton = document.getElementById("cancel-uninstall");
|
||||
const confirmButton = document.getElementById("confirm-uninstall")
|
||||
|
||||
uninstallOverlay.style.display = 'flex';
|
||||
document.body.classList.add('no-scroll');
|
||||
setTimeout(() => {
|
||||
uninstallOverlay.style.opacity = 1;
|
||||
uninstallContent.classList.add('open');
|
||||
}, 10)
|
||||
|
||||
const closeuninstallOverlay = () => {
|
||||
document.body.classList.remove('no-scroll');
|
||||
uninstallOverlay.style.opacity = 0;
|
||||
uninstallContent.classList.remove('open');
|
||||
setTimeout(() => {
|
||||
uninstallOverlay.style.display = 'none';
|
||||
}, 200)
|
||||
}
|
||||
cancelButton.addEventListener('click', () => closeuninstallOverlay());
|
||||
uninstallOverlay.addEventListener('click', (e) => {
|
||||
if (e.target === uninstallOverlay) closeuninstallOverlay();
|
||||
})
|
||||
confirmButton.addEventListener('click', () => {
|
||||
closeuninstallOverlay();
|
||||
uninstallWebUI();
|
||||
})
|
||||
});
|
||||
async function uninstallWebUI() {
|
||||
try {
|
||||
await execCommand(`sh ${basePath}common/get_extra.sh --uninstall`);
|
||||
console.log("uninstall script executed successfully.");
|
||||
await execCommand(`sh ${basePath}/common/get_extra.sh --uninstall`);
|
||||
showPrompt("prompt.uninstall_prompt");
|
||||
} catch (error) {
|
||||
console.error("Failed to execute uninstall command:", error);
|
||||
showPrompt("prompt.uninstall_failed", false);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Function to check if running in MMRL
|
||||
async function checkMMRL() {
|
||||
if (typeof ksu !== 'undefined' && ksu.mmrl) {
|
||||
// Adjust elements position for MMRL
|
||||
title.style.top = 'var(--window-inset-top)';
|
||||
const insetTop = getComputedStyle(document.documentElement).getPropertyValue('--window-inset-top');
|
||||
const insetTopValue = parseInt(insetTop, 10);
|
||||
searchMenuContainer.style.top = `${insetTopValue + 40}px`;
|
||||
headerBlock.style.display = 'block';
|
||||
floatingCard.style.bottom = 'calc(var(--window-inset-bottom) + 50px)';
|
||||
|
||||
// Set status bars theme based on device theme
|
||||
try {
|
||||
$tricky_store.setLightStatusBars(!window.matchMedia('(prefers-color-scheme: dark)').matches)
|
||||
} catch (error) {
|
||||
console.log("Error setting status bars theme:", error)
|
||||
console.error("Error setting status bars theme:", error)
|
||||
}
|
||||
|
||||
// Request API permission, supported version: 33045+
|
||||
try {
|
||||
$tricky_store.requestAdvancedKernelSUAPI();
|
||||
} catch (error) {
|
||||
console.log("Error requesting API:", error);
|
||||
console.error("Error requesting API:", error);
|
||||
}
|
||||
|
||||
// Check permissions
|
||||
@@ -214,19 +227,16 @@ async function checkMMRL() {
|
||||
MMRL_API = true;
|
||||
} catch (error) {
|
||||
console.error('Permission check failed:', error);
|
||||
permissionPopup.classList.remove('hidden');
|
||||
permissionPopup.style.display = 'flex';
|
||||
MMRL_API = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Funtion to adapt floating button hide in MMRL
|
||||
function hideFloatingBtn() {
|
||||
if (typeof ksu !== 'undefined' && ksu.mmrl) {
|
||||
floatingBtn.style.transform = 'translateY(calc(var(--window-inset-bottom) + 120px))';
|
||||
} else {
|
||||
floatingBtn.style.transform = 'translateY(120px)';
|
||||
}
|
||||
export function hideFloatingBtn(hide = true) {
|
||||
if (!hide) floatingCard.style.transform = 'translateY(0)';
|
||||
else floatingCard.style.transform = 'translateY(calc(var(--window-inset-bottom, 0px) + 120px))';
|
||||
}
|
||||
|
||||
// Function to apply ripple effect
|
||||
@@ -300,30 +310,28 @@ window.addEventListener('scroll', () => {
|
||||
}, 200);
|
||||
if (isRefreshing) return;
|
||||
if (window.scrollY > lastScrollY && window.scrollY > scrollThreshold) {
|
||||
title.style.transform = 'translateY(-80px)';
|
||||
headerBlock.style.transform = 'translateY(-80px)';
|
||||
title.style.transform = 'translateY(-100%)';
|
||||
searchMenuContainer.style.transform = 'translateY(-40px)';
|
||||
hideFloatingBtn();
|
||||
} else if (window.scrollY < lastScrollY) {
|
||||
headerBlock.style.transform = 'translateY(0)';
|
||||
title.style.transform = 'translateY(0)';
|
||||
searchMenuContainer.style.transform = 'translateY(0)';
|
||||
floatingBtn.style.transform = 'translateY(0)';
|
||||
hideFloatingBtn(false);
|
||||
}
|
||||
lastScrollY = window.scrollY;
|
||||
});
|
||||
|
||||
// Initial load
|
||||
document.addEventListener('DOMContentLoaded', async () => {
|
||||
await loadTranslations();
|
||||
await checkMMRL();
|
||||
if (!MMRL_API) return;
|
||||
await getBasePath();
|
||||
hideFloatingBtn();
|
||||
getModuleVersion();
|
||||
await initializeAvailableLanguages();
|
||||
const userLang = detectUserLanguage();
|
||||
await loadTranslations(userLang);
|
||||
setupMenuToggle();
|
||||
setupLanguageMenu();
|
||||
setupSystemAppMenu();
|
||||
await fetchAppList();
|
||||
applyRippleEffect();
|
||||
checkTrickyStoreVersion();
|
||||
@@ -331,12 +339,9 @@ document.addEventListener('DOMContentLoaded', async () => {
|
||||
updateCheck();
|
||||
securityPatch();
|
||||
loadingIndicator.style.display = "none";
|
||||
floatingBtn.style.opacity = '1';
|
||||
setTimeout(() => {
|
||||
floatingBtn.style.transform = 'translateY(0)';
|
||||
}, 10);
|
||||
floatingBtn.style.display = 'block';
|
||||
hideFloatingBtn(false);
|
||||
document.getElementById("refresh").addEventListener("click", refreshAppList);
|
||||
document.getElementById("aospkb").addEventListener("click", aospkb);
|
||||
document.querySelector('.uninstall-container').classList.remove('hidden-uninstall');
|
||||
});
|
||||
|
||||
@@ -364,5 +369,9 @@ export async function execCommand(command) {
|
||||
|
||||
// Function to toast message
|
||||
export function toast(message) {
|
||||
ksu.toast(message);
|
||||
try {
|
||||
ksu.toast(message);
|
||||
} catch (error) {
|
||||
console.error("Failed to show toast:", error);
|
||||
}
|
||||
}
|
||||
4
module/webui/scripts/marked.min.js
vendored
4
module/webui/scripts/marked.min.js
vendored
File diff suppressed because one or more lines are too long
@@ -1,4 +1,4 @@
|
||||
import { basePath, execCommand, showPrompt, toast, applyRippleEffect } from './main.js';
|
||||
import { basePath, execCommand, showPrompt, toast, applyRippleEffect, refreshAppList } from './main.js';
|
||||
|
||||
// Function to check or uncheck all app
|
||||
function toggleCheckboxes(shouldCheck) {
|
||||
@@ -42,11 +42,16 @@ document.getElementById("deselect-unnecessary").addEventListener("click", async
|
||||
try {
|
||||
const excludeList = await fetch("https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/more-exclude.json")
|
||||
.then(response => {
|
||||
if (!response.ok) {
|
||||
throw new Error(`HTTP error! status: ${response.status}`);
|
||||
}
|
||||
if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
|
||||
return response.json();
|
||||
})
|
||||
.catch(async () => {
|
||||
return fetch("https://raw.gitmirror.com/KOWX712/Tricky-Addon-Update-Target-List/main/more-exclude.json")
|
||||
.then(response => {
|
||||
if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
|
||||
return response.json();
|
||||
});
|
||||
})
|
||||
.then(data => {
|
||||
return data.data
|
||||
.flatMap(category => category.apps)
|
||||
@@ -57,7 +62,7 @@ document.getElementById("deselect-unnecessary").addEventListener("click", async
|
||||
toast("Failed to download unnecessary apps!");
|
||||
throw error;
|
||||
});
|
||||
const xposed = await execCommand(`sh ${basePath}common/get_extra.sh --xposed`);
|
||||
const xposed = await execCommand(`sh ${basePath}/common/get_extra.sh --xposed`);
|
||||
const UnnecessaryApps = excludeList.split("\n").map(app => app.trim()).filter(Boolean).concat(xposed.split("\n").map(app => app.trim()).filter(Boolean));
|
||||
const apps = document.querySelectorAll(".card");
|
||||
apps.forEach(app => {
|
||||
@@ -75,12 +80,112 @@ document.getElementById("deselect-unnecessary").addEventListener("click", async
|
||||
}
|
||||
});
|
||||
|
||||
// Function to add system app
|
||||
export async function setupSystemAppMenu() {
|
||||
document.getElementById("add-system-app").addEventListener("click", () => openSystemAppOverlay());
|
||||
document.getElementById("add-system-app-overlay").addEventListener("click", (event) => {
|
||||
if (event.target === event.currentTarget) closeSystemAppOverlay();
|
||||
});
|
||||
const systemAppOverlay = document.getElementById("add-system-app-overlay");
|
||||
const systemAppContent = document.querySelector('.add-system-app-card');
|
||||
const systemAppInput = document.getElementById("system-app-input");
|
||||
function openSystemAppOverlay() {
|
||||
renderSystemAppList();
|
||||
document.body.classList.add("no-scroll");
|
||||
systemAppOverlay.style.display = "flex";
|
||||
setTimeout(() => {
|
||||
systemAppOverlay.style.opacity = "1";
|
||||
systemAppContent.classList.add('open');
|
||||
}, 10);
|
||||
systemAppInput.value = "";
|
||||
}
|
||||
function closeSystemAppOverlay() {
|
||||
document.body.classList.remove("no-scroll");
|
||||
systemAppOverlay.style.opacity = "0";
|
||||
systemAppContent.classList.remove('open');
|
||||
setTimeout(() => {
|
||||
systemAppOverlay.style.display = "none";
|
||||
}, 300);
|
||||
}
|
||||
|
||||
// Add system app button
|
||||
document.getElementById("add-system-app-button").addEventListener("click", async () => {
|
||||
const input = document.getElementById("system-app-input");
|
||||
const packageName = input.value.trim();
|
||||
if (packageName) {
|
||||
try {
|
||||
const result = await execCommand(`pm list packages -s | grep -q ${packageName} || echo "false"`);
|
||||
if (result.includes("false")) {
|
||||
showPrompt("prompt.system_app_not_found", false);
|
||||
} else {
|
||||
await execCommand(`
|
||||
touch "/data/adb/tricky_store/system_app"
|
||||
echo "${packageName}" >> "/data/adb/tricky_store/system_app"
|
||||
echo "${packageName}" >> "/data/adb/tricky_store/target.txt"
|
||||
`);
|
||||
systemAppInput.value = "";
|
||||
closeSystemAppOverlay();
|
||||
refreshAppList();
|
||||
}
|
||||
} catch (error) {
|
||||
console.error("Error adding system app:", error);
|
||||
showPrompt("prompt.add_system_app_error", false);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Display current system app list and remove button
|
||||
async function renderSystemAppList() {
|
||||
const currentSystemAppList = document.querySelector(".current-system-app-list");
|
||||
const currentSystemAppListContent = document.querySelector(".current-system-app-list-content");
|
||||
currentSystemAppListContent.innerHTML = "";
|
||||
try {
|
||||
const systemAppList = await execCommand(`[ -f "/data/adb/tricky_store/system_app" ] && cat "/data/adb/tricky_store/system_app" | sed '/^$/d' || echo "false"`);
|
||||
if (systemAppList.trim() === 'false' || systemAppList.trim() === '') {
|
||||
currentSystemAppList.style.display = "none";
|
||||
} else {
|
||||
systemAppList.split("\n").forEach(app => {
|
||||
currentSystemAppListContent.innerHTML += `
|
||||
<div class="system-app-item">
|
||||
<span>${app}</span>
|
||||
<button class="remove-system-app-button ripple-element">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="22px" viewBox="0 -960 960 960" width="22px" fill="#FFFFFF"><path d="M154-412v-136h652v136H154Z"/></svg>
|
||||
</button>
|
||||
</div>
|
||||
`;
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
currentSystemAppList.style.display = "none";
|
||||
console.error("Error displaying system app list:", error);
|
||||
}
|
||||
|
||||
const removeSystemAppButtons = document.querySelectorAll(".remove-system-app-button");
|
||||
removeSystemAppButtons.forEach(button => {
|
||||
button.addEventListener("click", async () => {
|
||||
const app = button.closest(".system-app-item").querySelector("span").textContent;
|
||||
try {
|
||||
await execCommand(`
|
||||
sed -i "/${app}/d" "/data/adb/tricky_store/system_app" || true
|
||||
sed -i "/${app}/d" "/data/adb/tricky_store/target.txt" || true
|
||||
`);
|
||||
closeSystemAppOverlay();
|
||||
refreshAppList();
|
||||
} catch (error) {
|
||||
console.error("Error removing system app:", error);
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// Function to backup previous keybox and set new keybox
|
||||
async function setKeybox(content) {
|
||||
const sanitizedContent = content.replace(/'/g, "'\\''");
|
||||
try {
|
||||
await execCommand(`
|
||||
mv -f /data/adb/tricky_store/keybox.xml /data/adb/tricky_store/keybox.xml.bak 2>/dev/null
|
||||
echo '${content}' > /data/adb/tricky_store/keybox.xml
|
||||
echo '${sanitizedContent}' > /data/adb/tricky_store/keybox.xml
|
||||
chmod 644 /data/adb/tricky_store/keybox.xml
|
||||
`);
|
||||
return true;
|
||||
@@ -91,8 +196,8 @@ async function setKeybox(content) {
|
||||
}
|
||||
|
||||
// Function to replace aosp kb
|
||||
export async function aospkb() {
|
||||
const source = await execCommand(`xxd -r -p ${basePath}common/.default | base64 -d`);
|
||||
document.getElementById("aospkb").addEventListener("click", async () => {
|
||||
const source = await execCommand(`xxd -r -p ${basePath}/common/.default | base64 -d`);
|
||||
const result = await setKeybox(source);
|
||||
if (result) {
|
||||
console.log("AOSP keybox copied successfully.");
|
||||
@@ -100,81 +205,53 @@ export async function aospkb() {
|
||||
} else {
|
||||
showPrompt("prompt.key_set_error", false);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Function to replace valid kb
|
||||
document.getElementById("validkb").addEventListener("click", async () => {
|
||||
fetch("https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/.extra")
|
||||
.then(response => {
|
||||
if (!response.ok) {
|
||||
throw new Error(`HTTP error! status: ${response.status}`);
|
||||
}
|
||||
return response.text();
|
||||
})
|
||||
.then(async data => {
|
||||
if (!data.trim()) {
|
||||
await aospkb();
|
||||
showPrompt("prompt.no_valid_fallback", false);
|
||||
return;
|
||||
}
|
||||
try {
|
||||
const hexBytes = new Uint8Array(data.match(/.{1,2}/g).map(byte => parseInt(byte, 16)));
|
||||
const decodedHex = new TextDecoder().decode(hexBytes);
|
||||
const source = atob(decodedHex);
|
||||
const result = await setKeybox(source);
|
||||
if (result) {
|
||||
showPrompt("prompt.valid_key_set");
|
||||
} else {
|
||||
throw new Error("Failed to copy valid keybox");
|
||||
.then(response => {
|
||||
if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
|
||||
return response.text();
|
||||
})
|
||||
.catch(async () => {
|
||||
return fetch("https://raw.gitmirror.com/KOWX712/Tricky-Addon-Update-Target-List/main/.extra")
|
||||
.then(response => {
|
||||
if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
|
||||
return response.text();
|
||||
});
|
||||
})
|
||||
.then(async data => {
|
||||
if (!data.trim()) {
|
||||
await aospkb();
|
||||
showPrompt("prompt.no_valid_fallback", false);
|
||||
return;
|
||||
}
|
||||
} catch (error) {
|
||||
throw new Error("Failed to decode keybox data");
|
||||
}
|
||||
})
|
||||
.catch(async error => {
|
||||
showPrompt("prompt.no_internet", false);
|
||||
});
|
||||
try {
|
||||
const hexBytes = new Uint8Array(data.match(/.{1,2}/g).map(byte => parseInt(byte, 16)));
|
||||
const decodedHex = new TextDecoder().decode(hexBytes);
|
||||
const source = atob(decodedHex);
|
||||
const result = await setKeybox(source);
|
||||
if (result) {
|
||||
showPrompt("prompt.valid_key_set");
|
||||
} else {
|
||||
throw new Error("Failed to copy valid keybox");
|
||||
}
|
||||
} catch (error) {
|
||||
throw new Error("Failed to decode keybox data");
|
||||
}
|
||||
})
|
||||
.catch(async error => {
|
||||
showPrompt("prompt.no_internet", false);
|
||||
});
|
||||
});
|
||||
|
||||
// Add file selector dialog elements dynamically
|
||||
const fileSelector = document.createElement('div');
|
||||
fileSelector.className = 'file-selector-overlay';
|
||||
fileSelector.innerHTML = `
|
||||
<div class="file-selector">
|
||||
<div class="file-selector-header">
|
||||
<button class="back-button">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="20" viewBox="0 -960 960 960" width="20"><path d="M400-80 0-480l400-400 56 57-343 343 343 343-56 57Z"/></svg>
|
||||
</button>
|
||||
<div class="current-path">/storage/emulated/0/Download</div>
|
||||
<button class="close-selector">✕</button>
|
||||
</div>
|
||||
<div class="file-list"></div>
|
||||
</div>
|
||||
`;
|
||||
document.body.appendChild(fileSelector);
|
||||
|
||||
// Add styles for animations
|
||||
const style = document.createElement('style');
|
||||
style.textContent = `
|
||||
.file-selector-overlay {
|
||||
transition: opacity 0.3s ease;
|
||||
opacity: 0;
|
||||
}
|
||||
.file-selector-overlay.visible {
|
||||
opacity: 1;
|
||||
}
|
||||
.file-list {
|
||||
transition: transform 0.3s ease, opacity 0.3s ease;
|
||||
}
|
||||
.file-list.switching {
|
||||
transform: scale(0.95);
|
||||
opacity: 0;
|
||||
}
|
||||
`;
|
||||
document.head.appendChild(style);
|
||||
|
||||
// File selector
|
||||
const fileSelector = document.querySelector('.file-selector-overlay');
|
||||
const fileSelectorContent = document.querySelector('.file-selector');
|
||||
let currentPath = '/storage/emulated/0/Download';
|
||||
|
||||
// Function to display file in current path
|
||||
function updateCurrentPath() {
|
||||
const currentPathElement = document.querySelector('.current-path');
|
||||
const segments = currentPath.split('/').filter(Boolean);
|
||||
@@ -211,7 +288,7 @@ async function listFiles(path, skipAnimation = false) {
|
||||
// Add back button item if not in root directory
|
||||
if (currentPath !== '/storage/emulated/0') {
|
||||
const backItem = document.createElement('div');
|
||||
backItem.className = 'file-item';
|
||||
backItem.className = 'file-item ripple-element';
|
||||
backItem.innerHTML = `
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24">
|
||||
<path d="M141-160q-24 0-42-18.5T81-220v-520q0-23 18-41.5t42-18.5h280l60 60h340q23 0 41.5 18.5T881-680v460q0 23-18.5 41.5T821-160H141Z"/>
|
||||
@@ -219,23 +296,14 @@ async function listFiles(path, skipAnimation = false) {
|
||||
<span>..</span>
|
||||
`;
|
||||
backItem.addEventListener('click', async () => {
|
||||
currentPath = currentPath.split('/').slice(0, -1).join('/');
|
||||
if (currentPath === '') currentPath = '/storage/emulated/0';
|
||||
const currentPathElement = document.querySelector('.current-path');
|
||||
currentPathElement.innerHTML = currentPath.split('/').filter(Boolean).join('<span class="separator">›</span>');
|
||||
currentPathElement.scrollTo({
|
||||
left: currentPathElement.scrollWidth,
|
||||
behavior: 'smooth'
|
||||
});
|
||||
await listFiles(currentPath);
|
||||
document.querySelector('.back-button').click();
|
||||
});
|
||||
|
||||
fileList.appendChild(backItem);
|
||||
}
|
||||
items.forEach(item => {
|
||||
if (item.path === path) return;
|
||||
const itemElement = document.createElement('div');
|
||||
itemElement.className = 'file-item';
|
||||
itemElement.className = 'file-item ripple-element';
|
||||
itemElement.innerHTML = `
|
||||
<svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24">
|
||||
${item.isDirectory ?
|
||||
@@ -258,7 +326,7 @@ async function listFiles(path, skipAnimation = false) {
|
||||
const source = await execCommand(`cat "${item.path}"`);
|
||||
const result = await setKeybox(source);
|
||||
if (result) {
|
||||
fileSelector.style.display = 'none';
|
||||
closeCustomKeyboxSelector();
|
||||
showPrompt('prompt.custom_key_set');
|
||||
} else {
|
||||
showPrompt('prompt.custom_key_set_error');
|
||||
@@ -322,28 +390,32 @@ document.querySelector('.back-button').addEventListener('click', async () => {
|
||||
|
||||
// Close custom keybox selector
|
||||
document.querySelector('.close-selector').addEventListener('click', () => {
|
||||
fileSelector.classList.remove('visible');
|
||||
closeCustomKeyboxSelector();
|
||||
});
|
||||
fileSelector.addEventListener('click', (event) => {
|
||||
if (event.target === fileSelector) {
|
||||
closeCustomKeyboxSelector();
|
||||
}
|
||||
});
|
||||
|
||||
// Function to close custom keybox selector
|
||||
function closeCustomKeyboxSelector() {
|
||||
fileSelector.style.opacity = '0';
|
||||
fileSelectorContent.classList.remove('open');
|
||||
document.body.classList.remove("no-scroll");
|
||||
setTimeout(() => {
|
||||
fileSelector.style.display = 'none';
|
||||
}, 300);
|
||||
});
|
||||
fileSelector.addEventListener('click', (event) => {
|
||||
if (event.target === fileSelector) {
|
||||
fileSelector.classList.remove('visible');
|
||||
document.body.classList.remove("no-scroll");
|
||||
setTimeout(() => {
|
||||
fileSelector.style.display = 'none';
|
||||
}, 300);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Open custom keybox selector
|
||||
document.getElementById('customkb').addEventListener('click', async () => {
|
||||
fileSelector.style.display = 'flex';
|
||||
document.body.classList.add("no-scroll");
|
||||
fileSelector.offsetHeight;
|
||||
fileSelector.classList.add('visible');
|
||||
setTimeout(() => {
|
||||
fileSelector.style.opacity = '1';
|
||||
fileSelectorContent.classList.add('open');
|
||||
}, 10)
|
||||
currentPath = '/storage/emulated/0/Download';
|
||||
const currentPathElement = document.querySelector('.current-path');
|
||||
currentPathElement.innerHTML = currentPath.split('/').filter(Boolean).join('<span class="separator">›</span>');
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
import { basePath, execCommand, showPrompt } from './main.js';
|
||||
|
||||
const overlay = document.getElementById('security-patch-overlay');
|
||||
const card = document.getElementById('security-patch-card');
|
||||
const overlayContent = document.querySelector('.security-patch-card');
|
||||
const advancedToggle = document.getElementById('advanced-mode');
|
||||
const normalInputs = document.getElementById('normal-mode-inputs');
|
||||
const advancedInputs = document.getElementById('advanced-mode-inputs');
|
||||
@@ -16,11 +16,10 @@ const saveButton = document.getElementById('save-patch');
|
||||
// Show security patch dialog
|
||||
function showSecurityPatchDialog() {
|
||||
document.body.classList.add("no-scroll");
|
||||
overlay.style.display = 'block';
|
||||
card.style.display = 'block';
|
||||
overlay.style.display = 'flex';
|
||||
setTimeout(() => {
|
||||
overlay.style.opacity = '1';
|
||||
card.style.opacity = '1';
|
||||
overlayContent.classList.add('open');
|
||||
loadCurrentConfig();
|
||||
}, 10);
|
||||
}
|
||||
@@ -29,10 +28,9 @@ function showSecurityPatchDialog() {
|
||||
function hideSecurityPatchDialog() {
|
||||
document.body.classList.remove("no-scroll");
|
||||
overlay.style.opacity = '0';
|
||||
card.style.opacity = '0';
|
||||
overlayContent.classList.remove('open');
|
||||
setTimeout(() => {
|
||||
overlay.style.display = 'none';
|
||||
card.style.display = 'none';
|
||||
}, 200);
|
||||
}
|
||||
|
||||
@@ -41,7 +39,7 @@ async function handleSecurityPatch(mode, value = null) {
|
||||
if (mode === 'disable') {
|
||||
try {
|
||||
await execCommand(`
|
||||
sed -i "s/^auto_config=.*/auto_config=0/" /data/adb/security_patch
|
||||
rm -f /data/adb/tricky_store/security_patch_auto_config
|
||||
rm -f /data/adb/tricky_store/security_patch.txt
|
||||
`);
|
||||
showPrompt('security_patch.value_empty');
|
||||
@@ -53,7 +51,7 @@ async function handleSecurityPatch(mode, value = null) {
|
||||
} else if (mode === 'manual') {
|
||||
try {
|
||||
await execCommand(`
|
||||
sed -i "s/^auto_config=.*/auto_config=0/" /data/adb/security_patch
|
||||
rm -f /data/adb/tricky_store/security_patch_auto_config
|
||||
echo "${value}" > /data/adb/tricky_store/security_patch.txt
|
||||
chmod 644 /data/adb/tricky_store/security_patch.txt
|
||||
`);
|
||||
@@ -68,60 +66,47 @@ async function handleSecurityPatch(mode, value = null) {
|
||||
|
||||
// Load current configuration
|
||||
async function loadCurrentConfig() {
|
||||
let allValue, systemValue, bootValue, vendorValue;
|
||||
try {
|
||||
const result = await execCommand('cat /data/adb/security_patch');
|
||||
if (result) {
|
||||
const lines = result.split('\n');
|
||||
let autoConfig = '1', allValue = '0', systemValue = '0', bootValue = '0', vendorValue = '0';
|
||||
for (const line of lines) {
|
||||
if (line.startsWith('auto_config=')) {
|
||||
autoConfig = line.split('=')[1];
|
||||
}
|
||||
}
|
||||
|
||||
if (autoConfig === '1') {
|
||||
allValue = null;
|
||||
systemValue = null;
|
||||
bootValue = null;
|
||||
vendorValue = null;
|
||||
overlay.classList.add('hidden');
|
||||
} else {
|
||||
// Read values from tricky_store if auto_config is 0
|
||||
const trickyResult = await execCommand('cat /data/adb/tricky_store/security_patch.txt');
|
||||
if (trickyResult) {
|
||||
const trickyLines = trickyResult.split('\n');
|
||||
for (const line of trickyLines) {
|
||||
if (line.startsWith('all=')) {
|
||||
allValue = line.split('=')[1] || null;
|
||||
if (allValue !== null) allPatchInput.value = allValue;
|
||||
} else {
|
||||
allValue = null;
|
||||
}
|
||||
if (line.startsWith('system=')) {
|
||||
systemValue = line.split('=')[1] || null;
|
||||
if (systemValue !== null) systemPatchInput.value = systemValue;
|
||||
} else {
|
||||
systemValue = null;
|
||||
}
|
||||
if (line.startsWith('boot=')) {
|
||||
bootValue = line.split('=')[1] || null;
|
||||
if (bootValue !== null) bootPatchInput.value = bootValue;
|
||||
} else {
|
||||
bootValue = null;
|
||||
}
|
||||
if (line.startsWith('vendor=')) {
|
||||
vendorValue = line.split('=')[1] || null;
|
||||
if (vendorValue !== null) vendorPatchInput.value = vendorValue;
|
||||
} else {
|
||||
vendorValue = null;
|
||||
}
|
||||
const autoConfig = await execCommand('[ -f /data/adb/tricky_store/security_patch_auto_config ] && echo "true" || echo "false"');
|
||||
if (autoConfig.trim() === 'true') {
|
||||
allValue = null;
|
||||
systemValue = null;
|
||||
bootValue = null;
|
||||
vendorValue = null;
|
||||
} else {
|
||||
// Read values from tricky_store if auto_config is 0
|
||||
const trickyResult = await execCommand('cat /data/adb/tricky_store/security_patch.txt');
|
||||
if (trickyResult) {
|
||||
const trickyLines = trickyResult.split('\n');
|
||||
for (const line of trickyLines) {
|
||||
if (line.startsWith('all=')) {
|
||||
allValue = line.split('=')[1] || null;
|
||||
if (allValue !== null) allPatchInput.value = allValue;
|
||||
} else {
|
||||
allValue = null;
|
||||
}
|
||||
if (line.startsWith('system=')) {
|
||||
systemValue = line.split('=')[1] || null;
|
||||
if (systemValue !== null) systemPatchInput.value = systemValue;
|
||||
} else {
|
||||
systemValue = null;
|
||||
}
|
||||
if (line.startsWith('boot=')) {
|
||||
bootValue = line.split('=')[1] || null;
|
||||
if (bootValue !== null) bootPatchInput.value = bootValue;
|
||||
} else {
|
||||
bootValue = null;
|
||||
}
|
||||
if (line.startsWith('vendor=')) {
|
||||
vendorValue = line.split('=')[1] || null;
|
||||
if (vendorValue !== null) vendorPatchInput.value = vendorValue;
|
||||
} else {
|
||||
vendorValue = null;
|
||||
}
|
||||
}
|
||||
overlay.classList.add('hidden');
|
||||
}
|
||||
|
||||
// Check if in advanced mode
|
||||
if (autoConfig === '0' && allValue === null && (bootValue || systemValue || vendorValue)) {
|
||||
if (allValue === null && (bootValue || systemValue || vendorValue)) {
|
||||
checkAdvanced(true);
|
||||
}
|
||||
}
|
||||
@@ -240,11 +225,11 @@ export function securityPatch() {
|
||||
// Auto config button
|
||||
autoButton.addEventListener('click', async () => {
|
||||
try {
|
||||
const output = await execCommand(`sh ${basePath}common/get_extra.sh --security-patch`);
|
||||
const output = await execCommand(`sh ${basePath}/common/get_extra.sh --security-patch`);
|
||||
if (output.trim() === "not set") {
|
||||
showPrompt('security_patch.auto_failed', false);
|
||||
} else {
|
||||
await execCommand(`sed -i "s/^auto_config=.*/auto_config=1/" /data/adb/security_patch`);
|
||||
await execCommand(`touch /data/adb/tricky_store/security_patch_auto_config`);
|
||||
// Reset inputs
|
||||
allPatchInput.value = '';
|
||||
systemPatchInput.value = '';
|
||||
@@ -333,10 +318,11 @@ export function securityPatch() {
|
||||
try {
|
||||
showPrompt('security_patch.fetching');
|
||||
await new Promise(resolve => setTimeout(resolve, 200));
|
||||
const output = await execCommand(`sh ${basePath}common/get_extra.sh --get-security-patch`);
|
||||
const output = await execCommand(`sh ${basePath}/common/get_extra.sh --get-security-patch`);
|
||||
showPrompt('security_patch.fetched', true, 1000);
|
||||
checkAdvanced(true);
|
||||
|
||||
allPatchInput.value = output.replace(/-/g, '');
|
||||
systemPatchInput.value = 'prop';
|
||||
bootPatchInput.value = output;
|
||||
vendorPatchInput.value = output;
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
import { basePath, execCommand, showPrompt, noConnection, linkRedirect } from './main.js';
|
||||
import { updateCard } from './applist.js';
|
||||
|
||||
const updateCardText = document.getElementById('redirect-to-release');
|
||||
const UpdateMenu = document.querySelector('.update-overlay');
|
||||
const closeUpdate = document.querySelector('.close-update');
|
||||
const updateMenu = document.querySelector('.update-overlay');
|
||||
const updateMenuContent = document.querySelector('.update-menu');
|
||||
const closeUpdate = document.getElementById('close-update');
|
||||
const releaseNotes = document.querySelector('.changelog');
|
||||
const installButton = document.querySelector('.install');
|
||||
const rebootButton = document.querySelector('.reboot');
|
||||
@@ -26,7 +26,7 @@ function downloadFile(targetURL, fileName) {
|
||||
reader.onload = async function() {
|
||||
const base64Data = reader.result.split(',')[1];
|
||||
try {
|
||||
await execCommand(`echo ${base64Data} | base64 -d > ${basePath}common/tmp/${fileName}`);
|
||||
await execCommand(`echo ${base64Data} | base64 -d > ${basePath}/common/tmp/${fileName}`);
|
||||
resolve();
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
@@ -41,10 +41,11 @@ function downloadFile(targetURL, fileName) {
|
||||
// Function to check for updates
|
||||
export async function updateCheck() {
|
||||
try {
|
||||
const response = await fetch("https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/update.json");
|
||||
if (!response.ok) {
|
||||
throw new Error(`HTTP error! status: ${response.status}`);
|
||||
}
|
||||
const response = await fetch("https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/update.json")
|
||||
.catch(async () => {
|
||||
return fetch("https://raw.gitmirror.com/KOWX712/Tricky-Addon-Update-Target-List/main/update.json");
|
||||
});
|
||||
if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
|
||||
noConnection.style.display = "none";
|
||||
const data = await response.json();
|
||||
remoteVersionCode = data.versionCode;
|
||||
@@ -52,7 +53,7 @@ export async function updateCheck() {
|
||||
zipURL = data.zipUrl;
|
||||
changelogURL = data.changelog;
|
||||
|
||||
const updateAvailable = await execCommand(`sh ${basePath}common/get_extra.sh --check-update ${remoteVersionCode}`);
|
||||
const updateAvailable = await execCommand(`sh ${basePath}/common/get_extra.sh --check-update ${remoteVersionCode}`);
|
||||
if (updateAvailable.includes("update")) {
|
||||
showPrompt("prompt.new_update", true, 1500);
|
||||
updateCard.style.display = "flex";
|
||||
@@ -67,7 +68,7 @@ export async function updateCheck() {
|
||||
|
||||
// Function to render changelog
|
||||
async function renderChangelog() {
|
||||
const changelog = await execCommand(`sh ${basePath}common/get_extra.sh --release-note ${remoteVersion}`);
|
||||
const changelog = await execCommand(`sh ${basePath}/common/get_extra.sh --release-note ${remoteVersion}`);
|
||||
window.linkRedirect = linkRedirect;
|
||||
marked.setOptions({
|
||||
sanitize: true,
|
||||
@@ -84,24 +85,26 @@ async function renderChangelog() {
|
||||
.split('\n')
|
||||
.filter(line => line.trim() !== '')
|
||||
.join('\n');
|
||||
const formattedChangelog = marked.parse(cleanedChangelog);
|
||||
releaseNotes.innerHTML = formattedChangelog;
|
||||
const formattedChangelog = marked.parse(cleanedChangelog);
|
||||
releaseNotes.innerHTML = formattedChangelog;
|
||||
}
|
||||
|
||||
// Function to setup update menu
|
||||
function setupUpdateMenu() {
|
||||
function openUpdateMenu() {
|
||||
UpdateMenu.style.display = "flex";
|
||||
updateMenu.style.display = "flex";
|
||||
setTimeout(async () => {
|
||||
UpdateMenu.style.opacity = "1";
|
||||
updateMenu.style.opacity = "1";
|
||||
updateMenuContent.classList.add('open');
|
||||
}, 10);
|
||||
document.body.classList.add("no-scroll");
|
||||
}
|
||||
function closeUpdateMenu() {
|
||||
UpdateMenu.style.opacity = "0";
|
||||
updateMenu.style.opacity = "0";
|
||||
updateMenuContent.classList.remove('open');
|
||||
document.body.classList.remove("no-scroll");
|
||||
setTimeout(async () => {
|
||||
UpdateMenu.style.display = "none";
|
||||
updateMenu.style.display = "none";
|
||||
}, 200);
|
||||
}
|
||||
|
||||
@@ -109,8 +112,8 @@ function setupUpdateMenu() {
|
||||
updateCard.addEventListener('click', async () => {
|
||||
try {
|
||||
const module = await execCommand(`
|
||||
[ -f ${basePath}common/tmp/module.zip ] || echo "noModule"
|
||||
[ -f ${basePath}common/tmp/changelog.md ] || echo "noChangelog"
|
||||
[ -f ${basePath}/common/tmp/module.zip ] || echo "noModule"
|
||||
[ -f ${basePath}/common/tmp/changelog.md ] || echo "noChangelog"
|
||||
[ ! -f /data/adb/modules/TA_utl/update ] || echo "updated"
|
||||
`);
|
||||
if (module.trim().includes("updated")) {
|
||||
@@ -129,7 +132,7 @@ function setupUpdateMenu() {
|
||||
if (downloading) return;
|
||||
downloading = true;
|
||||
try {
|
||||
await execCommand(`sh ${basePath}common/get_extra.sh --get-update ${zipURL}`);
|
||||
await execCommand(`sh ${basePath}/common/get_extra.sh --get-update ${zipURL}`);
|
||||
showPrompt("prompt.downloaded");
|
||||
installButton.style.display = "flex";
|
||||
downloading = false;
|
||||
@@ -150,22 +153,19 @@ function setupUpdateMenu() {
|
||||
|
||||
// Close update menu
|
||||
closeUpdate.addEventListener("click", closeUpdateMenu);
|
||||
UpdateMenu.addEventListener("click", (event) => {
|
||||
if (event.target === UpdateMenu) {
|
||||
closeUpdateMenu();
|
||||
}
|
||||
updateMenu.addEventListener("click", (event) => {
|
||||
if (event.target === updateMenu) closeUpdateMenu();
|
||||
});
|
||||
|
||||
// Install button
|
||||
installButton.addEventListener('click', async () => {
|
||||
try {
|
||||
showPrompt("prompt.installing");
|
||||
setTimeout(async () => {
|
||||
await execCommand(`su -c 'sh ${basePath}common/get_extra.sh --install-update'`);
|
||||
showPrompt("prompt.installed");
|
||||
installButton.style.display = "none";
|
||||
rebootButton.style.display = "flex";
|
||||
}, 300);
|
||||
await new Promise(resolve => setTimeout(resolve, 300));
|
||||
await execCommand(`sh ${basePath}/common/get_extra.sh --install-update`);
|
||||
showPrompt("prompt.installed");
|
||||
installButton.style.display = "none";
|
||||
rebootButton.style.display = "flex";
|
||||
} catch (error) {
|
||||
showPrompt("prompt.install_fail", false);
|
||||
console.error('Fail to execute installation script:', error);
|
||||
@@ -176,9 +176,8 @@ function setupUpdateMenu() {
|
||||
rebootButton.addEventListener('click', async () => {
|
||||
try {
|
||||
showPrompt("prompt.rebooting");
|
||||
setTimeout(async () => {
|
||||
await execCommand("svc power reboot");
|
||||
}, 1000);
|
||||
await new Promise(resolve => setTimeout(resolve, 1000));
|
||||
await execCommand("svc power reboot");
|
||||
} catch (error) {
|
||||
showPrompt("prompt.reboot_fail", false);
|
||||
console.error('Fail to reboot:', error);
|
||||
|
||||
@@ -1,53 +1,24 @@
|
||||
.about-overlay {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
height: 100vh;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
z-index: 1100;
|
||||
display: none;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
opacity: 0;
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.about-menu {
|
||||
position: fixed;
|
||||
top: 50%;
|
||||
left: 50%;
|
||||
width: 90vw;
|
||||
position: relative;
|
||||
width: calc(90vw - 60px);
|
||||
max-width: 800px;
|
||||
transform: translate(-50%, -50%);
|
||||
background: #fff;
|
||||
background-color: var(--bg-secondary);
|
||||
border-radius: 15px;
|
||||
padding: 30px 0;
|
||||
z-index: 1200;
|
||||
padding: 30px;
|
||||
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.2);
|
||||
z-index: 1200;
|
||||
opacity: 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.close-about {
|
||||
position: absolute;
|
||||
top: 15px;
|
||||
right: 12px;
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 18px;
|
||||
color: #ccc;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.link,
|
||||
.about-content p {
|
||||
.about-menu p {
|
||||
margin: 0;
|
||||
padding: 0 30px;
|
||||
font-size: 16px;
|
||||
text-align: left;
|
||||
}
|
||||
@@ -69,13 +40,10 @@
|
||||
|
||||
#disclaimer {
|
||||
font-family: serif;
|
||||
width: calc(100% - 80px);
|
||||
width: calc(100% - 20px);
|
||||
padding: 8px 10px;
|
||||
left: 0;
|
||||
right: 0;
|
||||
margin: auto;
|
||||
border-radius: 10px;
|
||||
background-color: #F5F5F5;
|
||||
background-color: var(--border-color);
|
||||
}
|
||||
|
||||
#acknowledgment {
|
||||
@@ -120,14 +88,4 @@
|
||||
#link-text {
|
||||
font-size: 17px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.about-menu {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
#disclaimer {
|
||||
background-color: #6E6E6E;
|
||||
}
|
||||
}
|
||||
@@ -14,7 +14,7 @@
|
||||
flex-direction: column;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
background-color: #DCDCDC;
|
||||
background-color: var(--border-color);
|
||||
border: none;
|
||||
border-radius: 12px;
|
||||
box-sizing: border-box;
|
||||
@@ -22,7 +22,7 @@
|
||||
margin-bottom: 10px;
|
||||
outline: none;
|
||||
padding: 12px;
|
||||
width: calc(100% - 5px);
|
||||
width: calc(100% - 20px);
|
||||
max-width: 900px;
|
||||
}
|
||||
|
||||
@@ -43,40 +43,19 @@
|
||||
}
|
||||
|
||||
.update-overlay {
|
||||
display: none;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
height: 100vh;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
z-index: 1800;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
opacity: 0;
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.update-menu {
|
||||
position: relative;
|
||||
width: calc(90% - 60px);
|
||||
max-width: 800px;
|
||||
background-color: white;
|
||||
background-color: var(--bg-secondary);
|
||||
padding: 30px;
|
||||
border-radius: 15px;
|
||||
text-align: left;
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.3);
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.close-update {
|
||||
position: absolute;
|
||||
top: 15px;
|
||||
right: 12px;
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 20px;
|
||||
color: #ccc;
|
||||
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.2);
|
||||
}
|
||||
|
||||
.update-content h3 {
|
||||
@@ -101,12 +80,12 @@
|
||||
}
|
||||
|
||||
.changelog a {
|
||||
color: #6E6E6E;
|
||||
cursor: none;
|
||||
color: var(--text-secondary);
|
||||
cursor: default;
|
||||
}
|
||||
|
||||
.changelog a:active {
|
||||
color: blue;
|
||||
color: var(--btn-primary);
|
||||
}
|
||||
|
||||
.update-button-container {
|
||||
@@ -121,7 +100,8 @@
|
||||
display: none;
|
||||
justify-content: center;
|
||||
font-weight: bold;
|
||||
background-color: #ddd;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--border-color);
|
||||
width: 100%;
|
||||
border: none;
|
||||
padding: 12px;
|
||||
@@ -131,12 +111,12 @@
|
||||
}
|
||||
|
||||
.reboot {
|
||||
color: #fff;
|
||||
background-color: #007bff;
|
||||
color: var(--btn-primary-text);
|
||||
background-color: var(--btn-primary);
|
||||
}
|
||||
|
||||
.card {
|
||||
background-color: #fff;
|
||||
background-color: var(--bg-secondary);
|
||||
border: none;
|
||||
box-sizing: border-box;
|
||||
border-radius: 12px;
|
||||
@@ -144,7 +124,7 @@
|
||||
margin-bottom: 10px;
|
||||
outline: none;
|
||||
padding: 12px;
|
||||
width: calc(100% - 5px);
|
||||
width: calc(100% - 20px);
|
||||
max-width: 900px;
|
||||
transition: background-color 0.2s ease;
|
||||
}
|
||||
@@ -166,9 +146,8 @@
|
||||
right: 0;
|
||||
margin: auto;
|
||||
width: fit-content;
|
||||
background-color: #B1B1B1;
|
||||
background-color: var(--border-color);
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
||||
border: 1px solid #ccc;
|
||||
border-radius: 50px 50px;
|
||||
opacity: 0;
|
||||
transform: scale(0);
|
||||
@@ -217,7 +196,7 @@
|
||||
}
|
||||
|
||||
#normal-indicator {
|
||||
background-color: #007bff;
|
||||
background-color: var(--btn-primary);
|
||||
}
|
||||
|
||||
#hack-indicator {
|
||||
@@ -276,11 +255,15 @@
|
||||
transform: translate(-50%, -50%) scale(0);
|
||||
opacity: 0;
|
||||
transition: transform 0.2s ease-out, opacity 0.3s ease;
|
||||
|
||||
svg {
|
||||
fill: var(--btn-primary-text);
|
||||
}
|
||||
}
|
||||
|
||||
.checkbox:checked + .custom-checkbox {
|
||||
border-color: #007bff;
|
||||
background-color: #007bff;
|
||||
border-color: var(--btn-primary);
|
||||
background-color: var(--btn-primary);
|
||||
transition: border-color 0.1s ease;
|
||||
animation: checked-bounce 0.3s ease-out;
|
||||
}
|
||||
@@ -326,32 +309,4 @@
|
||||
100% {
|
||||
transform: scale(1);
|
||||
}
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.card {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.update-card {
|
||||
background-color: #4D4D4D;
|
||||
}
|
||||
|
||||
.mode {
|
||||
background-color: #343434;
|
||||
border: 1px solid #6E6E6E;
|
||||
}
|
||||
|
||||
.install {
|
||||
background-color: #6E6E6E;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.update-menu {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.update-content a {
|
||||
color: #C2C2C2;
|
||||
}
|
||||
}
|
||||
@@ -1,45 +1,20 @@
|
||||
.boot-hash-overlay {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
height: 100vh;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
z-index: 1200;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
opacity: 0;
|
||||
visibility: hidden;
|
||||
transition: opacity 0.2s ease, visibility 0.2s ease;
|
||||
}
|
||||
|
||||
.boot-hash-card {
|
||||
position: fixed;
|
||||
top: 10%;
|
||||
left: 50%;
|
||||
transform: translateX(-50%);
|
||||
width: calc(90% - 60px);
|
||||
max-width: 300px;
|
||||
background-color: #fff;
|
||||
height: fit-content;
|
||||
background-color: var(--bg-secondary);
|
||||
border-radius: 18px;
|
||||
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.2);
|
||||
z-index: 1200;
|
||||
padding: 30px;
|
||||
display: none;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
opacity: 0;
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.boot-hash-overlay.show {
|
||||
visibility: visible;
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
.boot-hash-card.show {
|
||||
display: flex;
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
.boot-hash-title {
|
||||
@@ -54,11 +29,12 @@
|
||||
height: 100px;
|
||||
font-size: 16px;
|
||||
padding: 10px;
|
||||
background-color: #F5F5F5;
|
||||
border: 1px solid #ccc;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-primary);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 10px;
|
||||
box-sizing: border-box;
|
||||
outline-color: #007bff;
|
||||
outline-color: var(--btn-primary);
|
||||
resize: none;
|
||||
}
|
||||
|
||||
@@ -69,21 +45,9 @@
|
||||
border-radius: 12px;
|
||||
font-size: 20px;
|
||||
font-weight: bold;
|
||||
background-color: #007bff;
|
||||
color: white;
|
||||
background-color: var(--btn-primary);
|
||||
color: var(--btn-primary-text);
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.boot-hash-card {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.boot-hash-input {
|
||||
background-color: #232323;
|
||||
color: #fff;
|
||||
border: 1px solid #6E6E6E;
|
||||
}
|
||||
}
|
||||
88
module/webui/styles/file_selector.css
Normal file
88
module/webui/styles/file_selector.css
Normal file
@@ -0,0 +1,88 @@
|
||||
.file-selector-overlay {
|
||||
z-index: 2000;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.file-selector {
|
||||
width: 90%;
|
||||
max-width: 600px;
|
||||
height: 80vh;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-secondary);
|
||||
border-radius: 15px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.file-selector-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 10px;
|
||||
border-bottom: 2px solid var(--border-color);
|
||||
}
|
||||
|
||||
.current-path .separator {
|
||||
color: var(--text-secondary);
|
||||
padding: 0 4px;
|
||||
}
|
||||
|
||||
.back-button {
|
||||
background: none;
|
||||
border: none;
|
||||
fill: var(--border-color);
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.current-path {
|
||||
flex-grow: 1;
|
||||
font-size: 16px;
|
||||
overflow: scroll;
|
||||
white-space: nowrap;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.close-selector {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 20px;
|
||||
color: var(--border-color);
|
||||
padding: 0 5px;
|
||||
}
|
||||
|
||||
.file-list {
|
||||
flex-grow: 1;
|
||||
overflow-y: auto;
|
||||
padding: 10px;
|
||||
transition: transform 0.3s ease, opacity 0.3s ease;
|
||||
}
|
||||
|
||||
.file-list.switching {
|
||||
transform: scale(0.95);
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.file-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 10px;
|
||||
border-radius: 8px;
|
||||
background-color: var(--bg-secondary);
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.file-item svg {
|
||||
flex-shrink: 0;
|
||||
margin-right: 10px;
|
||||
fill: var(--text-secondary);
|
||||
}
|
||||
|
||||
.file-item span {
|
||||
flex-grow: 1;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
@@ -1,7 +1,50 @@
|
||||
@import url('https://mui.kernelsu.org/mmrl/insets.css');
|
||||
@import url('https://mui.kernelsu.org/mmrl/colors.css');
|
||||
|
||||
:root {
|
||||
--top-inset: var(--window-inset-top, 0px);
|
||||
--bottom-inset: var(--window-inset-bottom, 0px);
|
||||
|
||||
/* Background colors */
|
||||
--bg-primary: var(--background, #F5F5F5);
|
||||
--bg-secondary: var(--tonalSurface, #fff);
|
||||
--bg-input: var(--surfaceBright, #F5F5F5);
|
||||
|
||||
/* Text colors */
|
||||
--text-primary: var(--onSurface, #000);
|
||||
--text-secondary: var(--onSurfaceVariant, #757575);
|
||||
|
||||
/* Border colors */
|
||||
--border-color: var(--outlineVariant, #ccc);
|
||||
|
||||
/* Button colors */
|
||||
--btn-primary: var(--primary, #007bff);
|
||||
--btn-primary-text: var(--onPrimary, #fff);
|
||||
--btn-uninstall: var(--error, #FF3636);
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
:root {
|
||||
/* Background colors */
|
||||
--bg-primary: var(--background, #151515);
|
||||
--bg-secondary: var(--tonalSurface, #292929);
|
||||
--bg-input: var(--surfaceBright, #1b1b1b);
|
||||
|
||||
/* Text colors */
|
||||
--text-primary: var(--onSurface, #fff);
|
||||
--text-secondary: var(--onSurfaceVariant, #C2C2C2);
|
||||
|
||||
/* Border colors */
|
||||
--border-color: var(--outlineVariant, #636363);
|
||||
}
|
||||
}
|
||||
|
||||
body {
|
||||
background-color: #F5F5F5;
|
||||
padding-top: var(--window-inset-top);
|
||||
padding-bottom: var(--window-inset-bottom);
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-primary);
|
||||
padding-top: var(--top-inset);
|
||||
padding-bottom: var(--bottom-inset);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.no-scroll {
|
||||
@@ -12,28 +55,60 @@ body {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
position: fixed;
|
||||
bottom: 50px;
|
||||
left: 50%;
|
||||
transform: translateX(-50%);
|
||||
z-index: 10;
|
||||
width: 100%;
|
||||
bottom: calc(var(--bottom-inset) + 50px);
|
||||
transition: transform 0.4s ease;
|
||||
pointer-events: none;
|
||||
z-index: 2;
|
||||
}
|
||||
|
||||
.floating-btn {
|
||||
flex-shrink: 0;
|
||||
background-color: #007bff;
|
||||
color: var(--btn-primary-text);
|
||||
background-color: var(--btn-primary);
|
||||
border: none;
|
||||
box-shadow: 0 4px 8px #0003;
|
||||
color: #fff;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
opacity: 0;
|
||||
display: none;
|
||||
bottom: 0;
|
||||
padding: 10px 20px;
|
||||
font-size: 22px;
|
||||
font-weight: bold;
|
||||
transition: transform 0.4s ease;
|
||||
border-radius: 50px 50px;
|
||||
user-select: none;
|
||||
pointer-events: auto;
|
||||
}
|
||||
|
||||
.overlay {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
height: 100vh;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
backdrop-filter: blur(5px);
|
||||
display: none;
|
||||
justify-content: center;
|
||||
opacity: 0;
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.overlay-content {
|
||||
transform: scale(0.8);
|
||||
transition: transform 0.2s ease;
|
||||
}
|
||||
|
||||
.overlay-content.open {
|
||||
transform: scale(1);
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
position: absolute;
|
||||
top: 12px;
|
||||
right: 12px;
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 18px;
|
||||
color: var(--border-color);
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.prompt {
|
||||
@@ -79,18 +154,22 @@ body {
|
||||
|
||||
.uninstall-container {
|
||||
padding: 8px;
|
||||
width: calc(100% - 5px);
|
||||
width: calc(100% - 20px);
|
||||
max-width: 900px;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
border-radius: 12px;
|
||||
border: 3px solid #FF3636;
|
||||
border: 3px solid var(--btn-uninstall);
|
||||
box-sizing: border-box;
|
||||
background-color: #F5F5F5;
|
||||
background-color: var(--bg-primary);
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
user-select: none;
|
||||
|
||||
svg {
|
||||
fill: var(--btn-uninstall);
|
||||
}
|
||||
}
|
||||
|
||||
.uninstall-container i {
|
||||
@@ -100,121 +179,73 @@ body {
|
||||
.uninstall-container span {
|
||||
font-size: 16px;
|
||||
font-weight: bold;
|
||||
color: #FF3636;
|
||||
color: var(--btn-uninstall);
|
||||
}
|
||||
|
||||
.uninstall-container.hidden-uninstall {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.file-selector-overlay {
|
||||
display: none;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
height: 100vh;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
z-index: 2000;
|
||||
justify-content: center;
|
||||
.uninstall-confirmation-overlay {
|
||||
align-items: center;
|
||||
z-index: 2000;
|
||||
}
|
||||
|
||||
.file-selector {
|
||||
.uninstall-confirmation {
|
||||
width: 90%;
|
||||
max-width: 600px;
|
||||
height: 80vh;
|
||||
background-color: #fff;
|
||||
max-height: 80%;
|
||||
overflow-y: auto;
|
||||
background-color: var(--bg-secondary);
|
||||
border-radius: 15px;
|
||||
padding: 30px;
|
||||
box-sizing: border-box;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.file-selector-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 10px;
|
||||
border-bottom: 1px solid #ccc;
|
||||
}
|
||||
|
||||
.current-path .separator {
|
||||
color: #6E6E6E;
|
||||
padding: 0 4px;
|
||||
}
|
||||
|
||||
.back-button {
|
||||
background: none;
|
||||
border: none;
|
||||
padding: 5px;
|
||||
margin-right: 10px;
|
||||
fill: #6E6E6E;
|
||||
.uninstall-confirmation p {
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.current-path {
|
||||
flex-grow: 1;
|
||||
font-size: 16px;
|
||||
overflow: scroll;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.close-selector {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 20px;
|
||||
color: #ccc;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.file-list {
|
||||
flex-grow: 1;
|
||||
overflow-y: auto;
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.file-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 10px;
|
||||
border-radius: 8px;
|
||||
background-color: #fff;
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
.uninstall-confirmation-title {
|
||||
font-size: 26px;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.file-item svg {
|
||||
margin-right: 10px;
|
||||
fill: #6E6E6E;
|
||||
.uninstall-confirmation-button-container {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
gap: 10px;
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
.file-item span {
|
||||
flex-grow: 1;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
.uninstall-confirmation-button {
|
||||
width: 100%;
|
||||
padding: 12px;
|
||||
border: none;
|
||||
border-radius: 12px;
|
||||
font-size: 18px;
|
||||
font-weight: bold;
|
||||
user-select: none;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--border-color);
|
||||
}
|
||||
|
||||
#confirm-uninstall {
|
||||
color: var(--btn-primary-text);
|
||||
background-color: var(--btn-primary);
|
||||
}
|
||||
|
||||
.permission-popup {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background-color: rgba(0, 0, 0, 0.85);
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
opacity: 1;
|
||||
align-items: center;
|
||||
z-index: 9999;
|
||||
}
|
||||
|
||||
.permission-popup.hidden {
|
||||
display: none;
|
||||
z-index: 2000;
|
||||
}
|
||||
|
||||
.permission-content {
|
||||
background-color: #fff;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-secondary);
|
||||
padding: 20px;
|
||||
border-radius: 12px;
|
||||
width: 80%;
|
||||
@@ -223,20 +254,20 @@ body {
|
||||
}
|
||||
|
||||
.permission-content h2 {
|
||||
color: #333;
|
||||
margin-bottom: 20px;
|
||||
font-size: 18px;
|
||||
}
|
||||
|
||||
.permission-steps {
|
||||
text-align: left;
|
||||
padding: 10px 20px;
|
||||
}
|
||||
|
||||
.permission-steps p {
|
||||
color: #333;
|
||||
margin: 10px 0;
|
||||
font-size: 16px;
|
||||
.permission-steps ol {
|
||||
padding: 0 25px;
|
||||
}
|
||||
|
||||
.permission-steps li {
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
.ripple-element {
|
||||
@@ -262,49 +293,4 @@ body {
|
||||
to {
|
||||
transform: scale(3);
|
||||
}
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
body {
|
||||
background-color: #121212;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.uninstall-container {
|
||||
background-color: #121212;
|
||||
}
|
||||
|
||||
.file-selector {
|
||||
background-color: #343434;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.file-selector-header {
|
||||
border-bottom: 1px solid #6E6E6E;
|
||||
}
|
||||
|
||||
.file-item {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.current-path .separator {
|
||||
color: #C2C2C2;
|
||||
}
|
||||
|
||||
.back-button {
|
||||
fill: #C2C2C2;
|
||||
}
|
||||
|
||||
.file-item svg {
|
||||
fill: #C2C2C2;
|
||||
}
|
||||
|
||||
.permission-content {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.permission-content h2,
|
||||
.permission-steps p {
|
||||
color: #fff;
|
||||
}
|
||||
}
|
||||
@@ -4,31 +4,21 @@
|
||||
justify-content: space-between;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
padding: 0 5px;
|
||||
padding-top: var(--top-inset);
|
||||
height: 40px;
|
||||
width: calc(100% - 10px);
|
||||
max-width: 1100px;
|
||||
background-color: #F5F5F5;
|
||||
background-color: var(--bg-primary);
|
||||
transition: transform 0.4s ease;
|
||||
z-index: 1100;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
left: 0;
|
||||
right: 0;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.header-block {
|
||||
background-color: #F5F5F5;
|
||||
display: none;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
z-index: 1100;
|
||||
transition: transform 0.4s ease;
|
||||
height: var(--window-inset-top);
|
||||
}
|
||||
|
||||
#module-version,
|
||||
#title {
|
||||
padding-left: 5px;
|
||||
@@ -56,7 +46,7 @@
|
||||
}
|
||||
|
||||
.language-icon {
|
||||
fill: #000;
|
||||
fill: var(--text-primary);
|
||||
}
|
||||
|
||||
.language-menu {
|
||||
@@ -64,10 +54,10 @@
|
||||
flex-direction: column;
|
||||
position: absolute;
|
||||
right: 5px;
|
||||
background-color: white;
|
||||
background-color: var(--bg-secondary);
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
||||
z-index: 1800;
|
||||
border: 1px solid #ccc;
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 8px;
|
||||
box-sizing: border-box;
|
||||
opacity: 0;
|
||||
@@ -84,10 +74,11 @@
|
||||
}
|
||||
|
||||
.language-option {
|
||||
flex-shrink: 0;
|
||||
padding: 8px 10px;
|
||||
text-align: left;
|
||||
color: #333;
|
||||
background-color: white;
|
||||
text-align: center;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-secondary);
|
||||
border: none;
|
||||
font-size: 16px;
|
||||
width: 100%;
|
||||
@@ -103,7 +94,7 @@
|
||||
left: 10px;
|
||||
width: calc(100% - 20px);
|
||||
height: 1px;
|
||||
background-color: #ccc;
|
||||
background-color: var(--border-color);
|
||||
}
|
||||
|
||||
.language-option:last-child::after {
|
||||
@@ -137,113 +128,38 @@
|
||||
}
|
||||
|
||||
.help-overlay {
|
||||
display: none;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
height: 100vh;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
z-index: 2000;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
opacity: 0;
|
||||
transition: opacity 0.4s ease;
|
||||
}
|
||||
|
||||
.help-overlay.show {
|
||||
display: flex;
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
.help-overlay.hide {
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.help-menu {
|
||||
position: relative;
|
||||
width: 90vw;
|
||||
width: calc(95vw - 60px);
|
||||
max-width: 800px;
|
||||
background-color: white;
|
||||
padding: 10px 0;
|
||||
background-color: var(--bg-secondary);
|
||||
padding: 30px;
|
||||
border-radius: 15px;
|
||||
text-align: left;
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.3);
|
||||
}
|
||||
|
||||
.close-help {
|
||||
position: absolute;
|
||||
top: 15px;
|
||||
right: 12px;
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 20px;
|
||||
color: #ccc;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.help-content {
|
||||
max-height: 85vh;
|
||||
padding: 0 30px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
max-height: calc(85vh - 60px);
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.help-content p {
|
||||
.help-content-header {
|
||||
font-size: 26px;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.help-content ul {
|
||||
padding-left: 0;
|
||||
list-style-type: none;
|
||||
}
|
||||
|
||||
.help-content ul li {
|
||||
font-weight: bold;
|
||||
.instruction strong {
|
||||
font-size: 18px;
|
||||
}
|
||||
|
||||
.help-content ul ul li {
|
||||
font-weight: normal;
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.help-content ul ul ul li {
|
||||
color: #777777;
|
||||
font-weight: normal;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.help-content ul ul ul li a {
|
||||
color: inherit;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.header-block,
|
||||
.header {
|
||||
background-color: #121212;
|
||||
}
|
||||
|
||||
.help-button {
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.language-icon {
|
||||
fill: #eee;
|
||||
}
|
||||
|
||||
.language-option,
|
||||
.help-menu {
|
||||
color: #eee;
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.language-menu {
|
||||
background-color: #343434;
|
||||
border: 1px solid #6E6E6E;
|
||||
}
|
||||
|
||||
.language-option::after {
|
||||
background-color: #6E6E6E;
|
||||
}
|
||||
.instruction p {
|
||||
margin: 0;
|
||||
}
|
||||
@@ -1,7 +1,7 @@
|
||||
.search-menu-container {
|
||||
display: flex;
|
||||
position: fixed;
|
||||
top: 40px;
|
||||
top: calc(var(--top-inset) + 40px);
|
||||
height: 50px;
|
||||
width: calc(100% - 20px);
|
||||
max-width: 1100px;
|
||||
@@ -14,8 +14,8 @@
|
||||
}
|
||||
|
||||
.search-card {
|
||||
background-color: white;
|
||||
border: 1px solid #ccc;
|
||||
background-color: var(--bg-secondary);
|
||||
border: 1px solid var(--border-color);
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
@@ -36,6 +36,8 @@
|
||||
.search-input {
|
||||
position: absolute;
|
||||
border: none;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-secondary);
|
||||
font-size: 17px;
|
||||
outline: none;
|
||||
left: 10px;
|
||||
@@ -45,7 +47,7 @@
|
||||
|
||||
.clear-btn {
|
||||
position: absolute;
|
||||
color: #ccc;
|
||||
color: var(--border-color);
|
||||
padding-bottom: 3px;
|
||||
right: 10px;
|
||||
border: none;
|
||||
@@ -68,8 +70,8 @@
|
||||
}
|
||||
|
||||
.menu-button {
|
||||
background-color: white;
|
||||
border: 1px solid #ccc;
|
||||
background-color: var(--bg-secondary);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 50%;
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
|
||||
width: 48px;
|
||||
@@ -81,7 +83,7 @@
|
||||
|
||||
.menu-icon {
|
||||
display: inline-block;
|
||||
fill: #000;
|
||||
fill: var(--text-primary);
|
||||
transform: rotate(0deg);
|
||||
transition: transform 0.2s ease;
|
||||
}
|
||||
@@ -91,8 +93,8 @@
|
||||
}
|
||||
|
||||
.menu-options {
|
||||
background-color: white;
|
||||
border: 1px solid #ccc;
|
||||
background-color: var(--bg-secondary);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 8px;
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
|
||||
box-sizing: border-box;
|
||||
@@ -131,7 +133,7 @@
|
||||
.menu-options li {
|
||||
padding: 12px 15px;
|
||||
text-align: left;
|
||||
background-color: white;
|
||||
background-color: var(--bg-secondary);
|
||||
}
|
||||
|
||||
.menu-options li::after {
|
||||
@@ -141,7 +143,7 @@
|
||||
left: 15px;
|
||||
width: calc(100% - 30px);
|
||||
height: 1px;
|
||||
background-color: #ccc;
|
||||
background-color: var(--border-color);
|
||||
}
|
||||
|
||||
.menu-options li:last-child::after {
|
||||
@@ -165,37 +167,4 @@
|
||||
background-color: none;
|
||||
z-index: 100;
|
||||
display: none;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.menu-icon {
|
||||
fill: #eee;
|
||||
}
|
||||
|
||||
.search-input,
|
||||
.search-card {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.search-card {
|
||||
border: 1px solid #6E6E6E;
|
||||
}
|
||||
|
||||
.search-input {
|
||||
color: white;
|
||||
}
|
||||
|
||||
.menu-options,
|
||||
#menu-button {
|
||||
background-color: #343434;
|
||||
border: 1px solid #6E6E6E;
|
||||
}
|
||||
|
||||
.menu-options li {
|
||||
background-color: #343434;
|
||||
}
|
||||
|
||||
.menu-options li::after {
|
||||
background-color: #6E6E6E
|
||||
}
|
||||
}
|
||||
@@ -1,33 +1,20 @@
|
||||
.security-patch-overlay {
|
||||
display: none;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background-color: rgba(0, 0, 0, 0.5);
|
||||
z-index: 2000;
|
||||
transition: opacity 0.2s ease;
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.security-patch-card {
|
||||
display: none;
|
||||
display: block;
|
||||
position: fixed;
|
||||
top: 10%;
|
||||
left: 50%;
|
||||
transform: translateX(-50%);
|
||||
background-color: white;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-secondary);
|
||||
padding: 30px;
|
||||
border-radius: 15px;
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
z-index: 2001;
|
||||
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.2);
|
||||
width: calc(90% - 60px);
|
||||
max-width: 300px;
|
||||
max-height: calc(80% - 60px);
|
||||
overflow-y: auto;
|
||||
transition: opacity 0.2s ease;
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.security-patch-content {
|
||||
@@ -71,7 +58,7 @@
|
||||
display: inline-block;
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
border: 2px solid #ccc;
|
||||
border: 2px solid var(--border-color);
|
||||
border-radius: 4px;
|
||||
box-sizing: border-box;
|
||||
transition: border-color 1s ease, transform 0.3s ease, background-color 0.4s ease;
|
||||
@@ -84,11 +71,15 @@
|
||||
transform: translate(-50%, -50%) scale(0);
|
||||
opacity: 0;
|
||||
transition: transform 0.2s ease-out, opacity 0.3s ease;
|
||||
|
||||
svg {
|
||||
fill: var(--btn-primary-text);
|
||||
}
|
||||
}
|
||||
|
||||
.advanced-toggle .checkbox:checked + .custom-checkbox {
|
||||
border-color: #007bff;
|
||||
background-color: #007bff;
|
||||
border-color: var(--btn-primary);
|
||||
background-color: var(--btn-primary);
|
||||
transition: border-color 0.1s ease;
|
||||
animation: checked-bounce 0.3s ease-out;
|
||||
}
|
||||
@@ -107,15 +98,16 @@
|
||||
.input-group label {
|
||||
padding-top: 10px;
|
||||
font-size: 14px;
|
||||
color: #666;
|
||||
color: var(--text-secondary);
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.input-group input {
|
||||
padding: 15px;
|
||||
background-color: #F5F5F5;
|
||||
border: 1px solid #ccc;
|
||||
outline-color: #007bff;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-primary);
|
||||
border: 1px solid var(--border-color);
|
||||
outline-color: var(--btn-primary);
|
||||
border-radius: 10px;
|
||||
font-size: 16px;
|
||||
}
|
||||
@@ -136,48 +128,21 @@
|
||||
border-radius: 12px;
|
||||
font-size: 18px;
|
||||
font-weight: bold;
|
||||
transition: background-color 0.2s ease;
|
||||
}
|
||||
|
||||
.get-button,
|
||||
.auto-button {
|
||||
background-color: #ddd;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--border-color);
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.save-button {
|
||||
background-color: #007bff;
|
||||
color: white;
|
||||
background-color: var(--btn-primary);
|
||||
color: var(--btn-primary-text);
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.hidden {
|
||||
display: none;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.security-patch-overlay {
|
||||
background-color: rgba(0, 0, 0, 0.8);
|
||||
}
|
||||
|
||||
.security-patch-card {
|
||||
background-color: #343434;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.input-group label {
|
||||
color: #ccc;
|
||||
}
|
||||
|
||||
.input-group input {
|
||||
background-color: #232323;
|
||||
color: #fff;
|
||||
border: 1px solid #6E6E6E;
|
||||
}
|
||||
|
||||
.get-button,
|
||||
.auto-button {
|
||||
background-color: #6E6E6E;
|
||||
color: white;
|
||||
}
|
||||
}
|
||||
}
|
||||
104
module/webui/styles/system_app.css
Normal file
104
module/webui/styles/system_app.css
Normal file
@@ -0,0 +1,104 @@
|
||||
.add-system-app-overlay {
|
||||
z-index: 2000;
|
||||
}
|
||||
|
||||
.add-system-app-card {
|
||||
position: fixed;
|
||||
display: flex;
|
||||
top: 10%;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
width: calc(90vw - 60px);
|
||||
max-width: 400px;
|
||||
max-height: calc(80vh - 60px);
|
||||
overflow-y: auto;
|
||||
padding: 30px;
|
||||
background-color: var(--bg-secondary);
|
||||
border-radius: 15px;
|
||||
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.2);
|
||||
}
|
||||
|
||||
.add-system-app-title {
|
||||
text-align: center;
|
||||
font-size: 26px;
|
||||
user-select: none;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.add-system-app-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.add-system-app-content input {
|
||||
width: 100%;
|
||||
padding: 15px;
|
||||
color: var(--text-primary);
|
||||
background-color: var(--bg-primary);
|
||||
border: 1px solid var(--border-color);
|
||||
outline-color: var(--btn-primary);
|
||||
border-radius: 10px;
|
||||
box-sizing: border-box;
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.add-system-app-button {
|
||||
margin-top: 15px;
|
||||
width: 100%;
|
||||
padding: 12px;
|
||||
border: none;
|
||||
border-radius: 12px;
|
||||
font-size: 18px;
|
||||
font-weight: bold;
|
||||
background-color: var(--btn-primary);
|
||||
color: var(--btn-primary-text);
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.current-system-app-list {
|
||||
text-align: center;
|
||||
margin-top: 25px;
|
||||
margin-bottom: 10px;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.current-system-app-list-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: flex-start;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.system-app-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
width: 100%;
|
||||
max-width: 100%;
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
padding: 10px;
|
||||
word-wrap: break-word;
|
||||
overflow-wrap: anywhere;
|
||||
word-break: break-all;
|
||||
}
|
||||
|
||||
.system-app-item:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.remove-system-app-button {
|
||||
flex-shrink: 0;
|
||||
margin-left: 10px;
|
||||
background-color: #e84d4d;
|
||||
border: none;
|
||||
border-radius: 8px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
}
|
||||
@@ -1,7 +1,5 @@
|
||||
{
|
||||
"description": "Unnecessary app list",
|
||||
"repo-link": "https://github.com/KOWX712/Tricky-Addon-Update-Target-List",
|
||||
"json-link": "https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/more-exclude.json",
|
||||
"data": [
|
||||
{
|
||||
"info": "Root manager",
|
||||
@@ -126,11 +124,5 @@
|
||||
}
|
||||
]
|
||||
}
|
||||
],
|
||||
"Add more app into this list?": [
|
||||
{
|
||||
"issue": "https://github.com/KOWX712/Tricky-Addon-Update-Target-List/issues",
|
||||
"pull-request": "https://github.com/KOWX712/Tricky-Addon-Update-Target-List/pulls"
|
||||
}
|
||||
]
|
||||
}
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"versionCode": 350,
|
||||
"version": "v3.5",
|
||||
"zipUrl": "https://github.com/KOWX712/Tricky-Addon-Update-Target-List/releases/download/v3.5/TrickyAddonModule-v3.5.zip",
|
||||
"versionCode": 380,
|
||||
"version": "v3.8",
|
||||
"zipUrl": "https://github.com/KOWX712/Tricky-Addon-Update-Target-List/releases/download/v3.8/TrickyAddonModule-v3.8.zip",
|
||||
"changelog": "https://raw.githubusercontent.com/KOWX712/Tricky-Addon-Update-Target-List/main/changelog.md"
|
||||
}
|
||||
Reference in New Issue
Block a user