Compare commits
	
		
			22 Commits
		
	
	
		
			appbase-v1
			...
			69187e3ed0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					69187e3ed0 | ||
| 
						 | 
					2e4003dae0 | ||
| 
						 | 
					1320984829 | ||
| 
						 | 
					abf1e5ba42 | ||
| 
						 | 
					1cd2f88038 | ||
| 
						 | 
					3f6e583d68 | ||
| 
						 | 
					271456bfcd | ||
| 
						 | 
					ee5458d82c | ||
| 
						 | 
					3a83367f71 | ||
| 
						 | 
					74b9350a6a | ||
| 
						 | 
					d2858f23f7 | ||
| 
						 | 
					40a5b9c339 | ||
| 
						 | 
					fd79113572 | ||
| 
						 | 
					9b911b583c | ||
| 
						 | 
					37817c3e8c | ||
| 
						 | 
					0b5402f5f3 | ||
| 
						 | 
					bea22e3853 | ||
| 
						 | 
					7e2ad0c01d | ||
| 
						 | 
					476ce02fc8 | ||
| 
						 | 
					bc697279ad | ||
| 
						 | 
					dee01f1179 | ||
| 
						 | 
					a500decc7a | 
@@ -113,10 +113,10 @@ if [[ $? -eq 0 ]]; then
 | 
				
			|||||||
    # 如果Git已经提交了所有代码就执行标签和应用发布操作
 | 
					    # 如果Git已经提交了所有代码就执行标签和应用发布操作
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # 预先询问是否添加工作流标签
 | 
					    # 预先询问是否添加工作流标签
 | 
				
			||||||
    echo "Add Github Workflows Tag? (yes/No)"
 | 
					    #echo "Add Github Workflows Tag? (yes/No)"
 | 
				
			||||||
	result=$(askAddWorkflowsTag)
 | 
						#result=$(askAddWorkflowsTag)
 | 
				
			||||||
	nAskAddWorkflowsTag=$?
 | 
						#nAskAddWorkflowsTag=$?
 | 
				
			||||||
	echo $result
 | 
						#echo $result
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # 发布应用
 | 
					    # 发布应用
 | 
				
			||||||
	echo "Publishing WinBoLL APK ..."
 | 
						echo "Publishing WinBoLL APK ..."
 | 
				
			||||||
@@ -138,17 +138,17 @@ if [[ $? -eq 0 ]]; then
 | 
				
			|||||||
	fi
 | 
						fi
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    # 添加 GitHub 工作流标签
 | 
					    # 添加 GitHub 工作流标签
 | 
				
			||||||
	if [[ $nAskAddWorkflowsTag -eq 1 ]]; then
 | 
						#if [[ $nAskAddWorkflowsTag -eq 1 ]]; then
 | 
				
			||||||
	    # 如果用户选择添加工作流标签
 | 
						    # 如果用户选择添加工作流标签
 | 
				
			||||||
    	result=$(addWorkflowsTag $1)
 | 
					    	#result=$(addWorkflowsTag $1)
 | 
				
			||||||
		if [[ $? -eq 0 ]]; then
 | 
							#if [[ $? -eq 0 ]]; then
 | 
				
			||||||
		    echo $result
 | 
							#    echo $result
 | 
				
			||||||
		    # 工作流标签添加成功
 | 
							    # 工作流标签添加成功
 | 
				
			||||||
		else
 | 
							#else
 | 
				
			||||||
			echo -e "${0}: addWorkflowsTag $1\n${result}\nAdd workflows tag cancel."
 | 
								#echo -e "${0}: addWorkflowsTag $1\n${result}\nAdd workflows tag cancel."
 | 
				
			||||||
			exit 1 # addWorkflowsTag 异常
 | 
								#exit 1 # addWorkflowsTag 异常
 | 
				
			||||||
		fi
 | 
							#fi
 | 
				
			||||||
	fi
 | 
						#fi
 | 
				
			||||||
	
 | 
						
 | 
				
			||||||
	## 清理更新描述文件内容
 | 
						## 清理更新描述文件内容
 | 
				
			||||||
	echo "" > $1/app_update_description.txt
 | 
						echo "" > $1/app_update_description.txt
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -114,9 +114,11 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
# 本项目要实际运用需要注意以下几个步骤:
 | 
					# 本项目要实际运用需要注意以下几个步骤:
 | 
				
			||||||
# 在项目根目录下:
 | 
					# 在项目根目录下:
 | 
				
			||||||
## 1. 项目模块编译环境设置(必须),settings.gradle-demo 要复制为 settings.gradle,并取消相应项目模块的注释。
 | 
					## ★. 项目模块编译环境设置(必须),settings.gradle-demo 要复制为 settings.gradle,并取消相应项目模块的注释。
 | 
				
			||||||
## 2. 项目 Android SDK 编译环境设置(可选),local.properties-demo 要复制为 local.properties,并按需要设置 Android SDK 目录。
 | 
					## ★. 项目 Android SDK 编译环境设置(可选),local.properties-demo 要复制为 local.properties,并按需要设置 Android SDK 目录。
 | 
				
			||||||
## 3. 类库型模块编译环境设置(可选),winboll.properties-demo 要复制为 winboll.properties,并按需要设置 WinBoLL Maven 库登录用户信息。
 | 
					## ★. 应用签名密钥 keystore 设置问题。一般调试编译只需用【Termux】cd 进 GenKeyStore 目录执行 $ bash gen_debug_keystore.sh 命令即可完成设置。
 | 
				
			||||||
 | 
					## ☆. 应用 WiBoLL 签名密钥配置问题<非必须考虑>。设置时需要 clone 【keystore】模块源码并拷贝模块目录的 appkey.jks 与 appkey.keystore 到项目根目录即可。
 | 
				
			||||||
 | 
					## ☆. 类库型模块编译环境设置(可选),winboll.properties-demo 要复制为 winboll.properties,并按需要设置 WinBoLL Maven 库登录用户信息。
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# ☆类库型项目编译方法
 | 
					# ☆类库型项目编译方法
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -29,7 +29,7 @@ android {
 | 
				
			|||||||
        // versionName 更新后需要手动设置 
 | 
					        // versionName 更新后需要手动设置 
 | 
				
			||||||
        // 项目模块目录的 build.gradle 文件的 stageCount=0
 | 
					        // 项目模块目录的 build.gradle 文件的 stageCount=0
 | 
				
			||||||
        // Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
 | 
					        // Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
 | 
				
			||||||
        versionName "15.8" 
 | 
					        versionName "15.9" 
 | 
				
			||||||
        if(true) {
 | 
					        if(true) {
 | 
				
			||||||
            versionName = genVersionName("${versionName}")
 | 
					            versionName = genVersionName("${versionName}")
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
#Created by .winboll/winboll_app_build.gradle
 | 
					#Created by .winboll/winboll_app_build.gradle
 | 
				
			||||||
#Tue Jun 03 19:17:05 HKT 2025
 | 
					#Mon Jun 09 01:44:28 HKT 2025
 | 
				
			||||||
stageCount=2
 | 
					stageCount=1
 | 
				
			||||||
libraryProject=libaes
 | 
					libraryProject=libaes
 | 
				
			||||||
baseVersion=15.8
 | 
					baseVersion=15.9
 | 
				
			||||||
publishVersion=15.8.1
 | 
					publishVersion=15.9.0
 | 
				
			||||||
buildCount=0
 | 
					buildCount=0
 | 
				
			||||||
baseBetaVersion=15.8.2
 | 
					baseBetaVersion=15.9.1
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -67,7 +67,7 @@ dependencies {
 | 
				
			|||||||
    //api 'androidx.vectordrawable:vectordrawable-animated:1.1.0'
 | 
					    //api 'androidx.vectordrawable:vectordrawable-animated:1.1.0'
 | 
				
			||||||
    //api 'androidx.fragment:fragment:1.1.0'
 | 
					    //api 'androidx.fragment:fragment:1.1.0'
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    api 'cc.winboll.studio:libaes:15.8.0'
 | 
					    api 'cc.winboll.studio:libaes:15.9.2'
 | 
				
			||||||
    api 'cc.winboll.studio:libapputils:15.8.2'
 | 
					    api 'cc.winboll.studio:libapputils:15.8.4'
 | 
				
			||||||
    api 'cc.winboll.studio:libappbase:15.8.2'
 | 
					    api 'cc.winboll.studio:libappbase:15.8.4'
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
#Created by .winboll/winboll_app_build.gradle
 | 
					#Created by .winboll/winboll_app_build.gradle
 | 
				
			||||||
#Sun Jun 01 08:03:56 GMT 2025
 | 
					#Sat Jun 28 05:02:54 GMT 2025
 | 
				
			||||||
stageCount=0
 | 
					stageCount=0
 | 
				
			||||||
libraryProject=
 | 
					libraryProject=
 | 
				
			||||||
baseVersion=15.0
 | 
					baseVersion=15.0
 | 
				
			||||||
publishVersion=15.0.0
 | 
					publishVersion=15.0.0
 | 
				
			||||||
buildCount=24
 | 
					buildCount=27
 | 
				
			||||||
baseBetaVersion=15.0.1
 | 
					baseBetaVersion=15.0.1
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
#Created by .winboll/winboll_app_build.gradle
 | 
					#Created by .winboll/winboll_app_build.gradle
 | 
				
			||||||
#Wed Jun 04 15:04:39 HKT 2025
 | 
					#Mon Jun 09 09:38:19 HKT 2025
 | 
				
			||||||
stageCount=8
 | 
					stageCount=9
 | 
				
			||||||
libraryProject=libappbase
 | 
					libraryProject=libappbase
 | 
				
			||||||
baseVersion=15.8
 | 
					baseVersion=15.8
 | 
				
			||||||
publishVersion=15.8.7
 | 
					publishVersion=15.8.8
 | 
				
			||||||
buildCount=0
 | 
					buildCount=0
 | 
				
			||||||
baseBetaVersion=15.8.8
 | 
					baseBetaVersion=15.8.9
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
#Created by .winboll/winboll_app_build.gradle
 | 
					#Created by .winboll/winboll_app_build.gradle
 | 
				
			||||||
#Tue Jun 03 19:16:41 HKT 2025
 | 
					#Mon Jun 09 01:44:28 HKT 2025
 | 
				
			||||||
stageCount=2
 | 
					stageCount=1
 | 
				
			||||||
libraryProject=libaes
 | 
					libraryProject=libaes
 | 
				
			||||||
baseVersion=15.8
 | 
					baseVersion=15.9
 | 
				
			||||||
publishVersion=15.8.1
 | 
					publishVersion=15.9.0
 | 
				
			||||||
buildCount=0
 | 
					buildCount=0
 | 
				
			||||||
baseBetaVersion=15.8.2
 | 
					baseBetaVersion=15.9.1
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -24,5 +24,7 @@ dependencies {
 | 
				
			|||||||
    api fileTree(dir: 'libs', include: ['*.jar'])
 | 
					    api fileTree(dir: 'libs', include: ['*.jar'])
 | 
				
			||||||
    // 网络连接类库
 | 
					    // 网络连接类库
 | 
				
			||||||
    api 'com.squareup.okhttp3:okhttp:4.4.1'
 | 
					    api 'com.squareup.okhttp3:okhttp:4.4.1'
 | 
				
			||||||
 | 
					    // https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-databind
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
 | 
					    api 'com.google.code.gson:gson:2.10.1'
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,8 +1,8 @@
 | 
				
			|||||||
#Created by .winboll/winboll_app_build.gradle
 | 
					#Created by .winboll/winboll_app_build.gradle
 | 
				
			||||||
#Wed Jun 04 15:04:39 HKT 2025
 | 
					#Mon Jun 09 09:38:19 HKT 2025
 | 
				
			||||||
stageCount=8
 | 
					stageCount=9
 | 
				
			||||||
libraryProject=libappbase
 | 
					libraryProject=libappbase
 | 
				
			||||||
baseVersion=15.8
 | 
					baseVersion=15.8
 | 
				
			||||||
publishVersion=15.8.7
 | 
					publishVersion=15.8.8
 | 
				
			||||||
buildCount=0
 | 
					buildCount=0
 | 
				
			||||||
baseBetaVersion=15.8.8
 | 
					baseBetaVersion=15.8.9
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -3,10 +3,14 @@ package cc.winboll.studio.libappbase.activities;
 | 
				
			|||||||
import android.app.Activity;
 | 
					import android.app.Activity;
 | 
				
			||||||
import android.os.Bundle;
 | 
					import android.os.Bundle;
 | 
				
			||||||
import android.view.View;
 | 
					import android.view.View;
 | 
				
			||||||
 | 
					import android.widget.RadioButton;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.BuildConfig;
 | 
				
			||||||
import cc.winboll.studio.libappbase.LogUtils;
 | 
					import cc.winboll.studio.libappbase.LogUtils;
 | 
				
			||||||
import cc.winboll.studio.libappbase.LogView;
 | 
					import cc.winboll.studio.libappbase.LogView;
 | 
				
			||||||
import cc.winboll.studio.libappbase.R;
 | 
					import cc.winboll.studio.libappbase.R;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.models.UserInfoModel;
 | 
				
			||||||
import cc.winboll.studio.libappbase.utils.RSAUtils;
 | 
					import cc.winboll.studio.libappbase.utils.RSAUtils;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.utils.YunUtils;
 | 
				
			||||||
import cc.winboll.studio.libappbase.winboll.IWinBoLLActivity;
 | 
					import cc.winboll.studio.libappbase.winboll.IWinBoLLActivity;
 | 
				
			||||||
import java.security.KeyPair;
 | 
					import java.security.KeyPair;
 | 
				
			||||||
import java.security.PrivateKey;
 | 
					import java.security.PrivateKey;
 | 
				
			||||||
@@ -21,6 +25,13 @@ public class LogonActivity extends Activity implements IWinBoLLActivity {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    public static final String TAG = "LogonActivity";
 | 
					    public static final String TAG = "LogonActivity";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static final String DEBUG_HOST = "http://10.8.0.250:456";
 | 
				
			||||||
 | 
					    public static final String YUN_HOST = "https://yun.winboll.cc";
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    String mHost = "";
 | 
				
			||||||
 | 
					    RadioButton mrbYunHost;
 | 
				
			||||||
 | 
					    RadioButton mrbDebugHost;
 | 
				
			||||||
    LogView mLogView;
 | 
					    LogView mLogView;
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
    @Override
 | 
					    @Override
 | 
				
			||||||
@@ -40,6 +51,25 @@ public class LogonActivity extends Activity implements IWinBoLLActivity {
 | 
				
			|||||||
        mLogView = findViewById(R.id.logview);
 | 
					        mLogView = findViewById(R.id.logview);
 | 
				
			||||||
        mLogView.start();
 | 
					        mLogView.start();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mHost = BuildConfig.DEBUG ? DEBUG_HOST: YUN_HOST;
 | 
				
			||||||
 | 
					        if (BuildConfig.DEBUG) {
 | 
				
			||||||
 | 
					            mrbYunHost = findViewById(R.id.rb_yunhost);
 | 
				
			||||||
 | 
					            mrbDebugHost = findViewById(R.id.rb_debughost);
 | 
				
			||||||
 | 
					            mrbYunHost.setChecked(!BuildConfig.DEBUG);
 | 
				
			||||||
 | 
					            mrbDebugHost.setChecked(BuildConfig.DEBUG);
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            findViewById(R.id.ll_hostbar).setVisibility(View.GONE);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    public void onSwitchHost(View view) {
 | 
				
			||||||
 | 
					        if (view.getId() == R.id.rb_yunhost) {
 | 
				
			||||||
 | 
					            mrbDebugHost.setChecked(false);
 | 
				
			||||||
 | 
					            mHost = YUN_HOST;
 | 
				
			||||||
 | 
					        } else if (view.getId() == R.id.rb_debughost) {
 | 
				
			||||||
 | 
					            mrbYunHost.setChecked(false);
 | 
				
			||||||
 | 
					            mHost = DEBUG_HOST;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Override
 | 
					    @Override
 | 
				
			||||||
@@ -48,24 +78,33 @@ public class LogonActivity extends Activity implements IWinBoLLActivity {
 | 
				
			|||||||
        mLogView.start();
 | 
					        mLogView.start();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void onTestLogin(View view) {
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "onTestLogin");
 | 
				
			||||||
 | 
					        final YunUtils yunUtils = YunUtils.getInstance(this);
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					        UserInfoModel userInfoModel = new UserInfoModel();
 | 
				
			||||||
 | 
					        userInfoModel.setUsername("jian");
 | 
				
			||||||
 | 
					        userInfoModel.setPassword("kkiio");
 | 
				
			||||||
 | 
					        userInfoModel.setToken("aaa111");
 | 
				
			||||||
 | 
					        yunUtils.login(mHost, userInfoModel);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void onTestRSA(View view) {
 | 
					    public void onTestRSA(View view) {
 | 
				
			||||||
        LogUtils.d(TAG, "onTestRSA");
 | 
					        LogUtils.d(TAG, "onTestRSA");
 | 
				
			||||||
        String keyPath = getFilesDir() + "/home/keys/"; // 密钥保存路径
 | 
					        RSAUtils utils = RSAUtils.getInstance(this);
 | 
				
			||||||
        LogUtils.d(TAG, String.format("keyPath %s", keyPath));
 | 
					 | 
				
			||||||
        RSAUtils utils = RSAUtils.getInstance();
 | 
					 | 
				
			||||||
        
 | 
					        
 | 
				
			||||||
        try {
 | 
					        try {
 | 
				
			||||||
            // 测试 1:首次生成密钥对
 | 
					            // 测试 1:首次生成密钥对
 | 
				
			||||||
            LogUtils.d(TAG, "==== 首次生成密钥对 ====");
 | 
					            LogUtils.d(TAG, "==== 首次生成密钥对 ====");
 | 
				
			||||||
            if (utils.keysExist(keyPath)) {
 | 
					            if (utils.keysExist()) {
 | 
				
			||||||
                LogUtils.d(TAG, "密钥对已生成");
 | 
					                LogUtils.d(TAG, "密钥对已生成");
 | 
				
			||||||
            } else {
 | 
					            } else {
 | 
				
			||||||
                utils.generateAndSaveKeys(keyPath);
 | 
					                utils.generateAndSaveKeys();
 | 
				
			||||||
                LogUtils.d(TAG, "密钥对生成成功,保存至:" + keyPath);
 | 
					                LogUtils.d(TAG, "密钥对生成成功。");
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            // 测试 2:获取密钥对(自动读取已生成的文件)
 | 
					            // 测试 2:获取密钥对(自动读取已生成的文件)
 | 
				
			||||||
            KeyPair keyPair = utils.getOrGenerateKeys(keyPath);
 | 
					            KeyPair keyPair = utils.getOrGenerateKeys();
 | 
				
			||||||
            PublicKey publicKey = keyPair.getPublic();
 | 
					            PublicKey publicKey = keyPair.getPublic();
 | 
				
			||||||
            PrivateKey privateKey = keyPair.getPrivate();
 | 
					            PrivateKey privateKey = keyPair.getPrivate();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -78,7 +117,7 @@ public class LogonActivity extends Activity implements IWinBoLLActivity {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            // 测试 3:重复调用时检查是否复用文件
 | 
					            // 测试 3:重复调用时检查是否复用文件
 | 
				
			||||||
            LogUtils.d(TAG, "\n==== 二次调用 ====");
 | 
					            LogUtils.d(TAG, "\n==== 二次调用 ====");
 | 
				
			||||||
            KeyPair reusedPair = utils.getOrGenerateKeys(keyPath);
 | 
					            KeyPair reusedPair = utils.getOrGenerateKeys();
 | 
				
			||||||
            LogUtils.d(TAG, "是否为同一公钥:" + (publicKey.equals(reusedPair.getPublic()))); // true(单例引用)
 | 
					            LogUtils.d(TAG, "是否为同一公钥:" + (publicKey.equals(reusedPair.getPublic()))); // true(单例引用)
 | 
				
			||||||
            LogUtils.d(TAG, "操作完成");
 | 
					            LogUtils.d(TAG, "操作完成");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -94,7 +133,7 @@ public class LogonActivity extends Activity implements IWinBoLLActivity {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            // 3. 私钥解密
 | 
					            // 3. 私钥解密
 | 
				
			||||||
            String decryptedMessage = utils.decryptWithPrivateKey(encryptedData, privateKeyReused);
 | 
					            String decryptedMessage = utils.decryptWithPrivateKey(encryptedData, privateKeyReused);
 | 
				
			||||||
            LogUtils.d(TAG, "解密结果:" + decryptedMessage);
 | 
					            LogUtils.d(TAG, "解密结果: " + decryptedMessage);
 | 
				
			||||||
            
 | 
					            
 | 
				
			||||||
            // 4. 验证解密是否成功
 | 
					            // 4. 验证解密是否成功
 | 
				
			||||||
            if (testMessage.equals(decryptedMessage)) {
 | 
					            if (testMessage.equals(decryptedMessage)) {
 | 
				
			||||||
@@ -106,4 +145,6 @@ public class LogonActivity extends Activity implements IWinBoLLActivity {
 | 
				
			|||||||
            LogUtils.d(TAG, e, Thread.currentThread().getStackTrace());
 | 
					            LogUtils.d(TAG, e, Thread.currentThread().getStackTrace());
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -17,8 +17,6 @@ public class APPModel extends BaseBean {
 | 
				
			|||||||
    // 应用是否处于正在调试状态
 | 
					    // 应用是否处于正在调试状态
 | 
				
			||||||
    //
 | 
					    //
 | 
				
			||||||
    boolean isDebuging = false;
 | 
					    boolean isDebuging = false;
 | 
				
			||||||
    // 用本机 RSA 加密后保存的令牌
 | 
					 | 
				
			||||||
    String rsaToken = "";
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public APPModel() {
 | 
					    public APPModel() {
 | 
				
			||||||
        this.isDebuging = false;
 | 
					        this.isDebuging = false;
 | 
				
			||||||
@@ -28,14 +26,6 @@ public class APPModel extends BaseBean {
 | 
				
			|||||||
        this.isDebuging = isDebuging;
 | 
					        this.isDebuging = isDebuging;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public void setRsaToken(String rsaToken) {
 | 
					 | 
				
			||||||
        this.rsaToken = rsaToken;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    public String getRsaToken() {
 | 
					 | 
				
			||||||
        return rsaToken;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    public void setIsDebuging(boolean isDebuging) {
 | 
					    public void setIsDebuging(boolean isDebuging) {
 | 
				
			||||||
        this.isDebuging = isDebuging;
 | 
					        this.isDebuging = isDebuging;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -53,7 +43,6 @@ public class APPModel extends BaseBean {
 | 
				
			|||||||
    public void writeThisToJsonWriter(JsonWriter jsonWriter) throws IOException {
 | 
					    public void writeThisToJsonWriter(JsonWriter jsonWriter) throws IOException {
 | 
				
			||||||
        super.writeThisToJsonWriter(jsonWriter);
 | 
					        super.writeThisToJsonWriter(jsonWriter);
 | 
				
			||||||
        jsonWriter.name("isDebuging").value(isDebuging());
 | 
					        jsonWriter.name("isDebuging").value(isDebuging());
 | 
				
			||||||
        jsonWriter.name("rsaToken").value(getRsaToken());
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @Override
 | 
					    @Override
 | 
				
			||||||
@@ -61,8 +50,6 @@ public class APPModel extends BaseBean {
 | 
				
			|||||||
        if (super.initObjectsFromJsonReader(jsonReader, name)) { return true; } else {
 | 
					        if (super.initObjectsFromJsonReader(jsonReader, name)) { return true; } else {
 | 
				
			||||||
            if (name.equals("isDebuging")) {
 | 
					            if (name.equals("isDebuging")) {
 | 
				
			||||||
                setIsDebuging(jsonReader.nextBoolean());
 | 
					                setIsDebuging(jsonReader.nextBoolean());
 | 
				
			||||||
            } else if (name.equals("rsaToken")) {
 | 
					 | 
				
			||||||
                setRsaToken(jsonReader.nextString());
 | 
					 | 
				
			||||||
            } else {
 | 
					            } else {
 | 
				
			||||||
                return false;
 | 
					                return false;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1,53 @@
 | 
				
			|||||||
 | 
					package cc.winboll.studio.libappbase.models;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * @Author ZhanGSKen<zhangsken@188.com>
 | 
				
			||||||
 | 
					 * @Date 2025/06/05 11:26
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class ResponseData {
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    public static final String STATUS_SUCCESS = "success";
 | 
				
			||||||
 | 
					    public static final String STATUS_ERROR = "error";
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    private String status;
 | 
				
			||||||
 | 
					    private String message;
 | 
				
			||||||
 | 
					    private UserInfoModel data;
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    public ResponseData() {
 | 
				
			||||||
 | 
					        this.status = "";
 | 
				
			||||||
 | 
					        this.message = "";
 | 
				
			||||||
 | 
					        this.data = new UserInfoModel();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    public ResponseData(String status, String message, UserInfoModel data) {
 | 
				
			||||||
 | 
					        this.status = status;
 | 
				
			||||||
 | 
					        this.message = message;
 | 
				
			||||||
 | 
					        this.data = data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setStatus(String status) {
 | 
				
			||||||
 | 
					        this.status = status;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public String getStatus() {
 | 
				
			||||||
 | 
					        return status;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setMessage(String message) {
 | 
				
			||||||
 | 
					        this.message = message;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public String getMessage() {
 | 
				
			||||||
 | 
					        return message;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setData(UserInfoModel data) {
 | 
				
			||||||
 | 
					        this.data = data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public UserInfoModel getData() {
 | 
				
			||||||
 | 
					        return data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -0,0 +1,92 @@
 | 
				
			|||||||
 | 
					package cc.winboll.studio.libappbase.models;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * @Author ZhanGSKen<zhangsken@188.com>
 | 
				
			||||||
 | 
					 * @Date 2025/06/04 19:14
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					import android.util.JsonReader;
 | 
				
			||||||
 | 
					import android.util.JsonWriter;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.BaseBean;
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class UserInfoModel extends BaseBean {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static final String TAG = "UserInfoModel";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    String username;
 | 
				
			||||||
 | 
					    String password;
 | 
				
			||||||
 | 
					    String token;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public UserInfoModel() {
 | 
				
			||||||
 | 
					        this.username = "";
 | 
				
			||||||
 | 
					        this.password = "";
 | 
				
			||||||
 | 
					        this.token = "";
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setUsername(String username) {
 | 
				
			||||||
 | 
					        this.username = username;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public String getUsername() {
 | 
				
			||||||
 | 
					        return username;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setPassword(String password) {
 | 
				
			||||||
 | 
					        this.password = password;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public String getPassword() {
 | 
				
			||||||
 | 
					        return password;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void setToken(String token) {
 | 
				
			||||||
 | 
					        this.token = token;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public String getToken() {
 | 
				
			||||||
 | 
					        return token;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    public String getName() {
 | 
				
			||||||
 | 
					        return UserInfoModel.class.getName();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    public void writeThisToJsonWriter(JsonWriter jsonWriter) throws IOException {
 | 
				
			||||||
 | 
					        super.writeThisToJsonWriter(jsonWriter);
 | 
				
			||||||
 | 
					        jsonWriter.name("username").value(getUsername());
 | 
				
			||||||
 | 
					        jsonWriter.name("password").value(getPassword());
 | 
				
			||||||
 | 
					        jsonWriter.name("token").value(getToken());
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    public boolean initObjectsFromJsonReader(JsonReader jsonReader, String name) throws IOException {
 | 
				
			||||||
 | 
					        if (super.initObjectsFromJsonReader(jsonReader, name)) { return true; } else {
 | 
				
			||||||
 | 
					            if (name.equals("username")) {
 | 
				
			||||||
 | 
					                setUsername(jsonReader.nextString());
 | 
				
			||||||
 | 
					            } else if (name.equals("password")) {
 | 
				
			||||||
 | 
					                setPassword(jsonReader.nextString());
 | 
				
			||||||
 | 
					            } else if (name.equals("token")) {
 | 
				
			||||||
 | 
					                setToken(jsonReader.nextString());
 | 
				
			||||||
 | 
					            } else {
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return true;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    public BaseBean readBeanFromJsonReader(JsonReader jsonReader) throws IOException {
 | 
				
			||||||
 | 
					        jsonReader.beginObject();
 | 
				
			||||||
 | 
					        while (jsonReader.hasNext()) {
 | 
				
			||||||
 | 
					            String name = jsonReader.nextName();
 | 
				
			||||||
 | 
					            if (!initObjectsFromJsonReader(jsonReader, name)) {
 | 
				
			||||||
 | 
					                jsonReader.skipValue();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        // 结束 JSON 对象
 | 
				
			||||||
 | 
					        jsonReader.endObject();
 | 
				
			||||||
 | 
					        return this;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -0,0 +1,128 @@
 | 
				
			|||||||
 | 
					package cc.winboll.studio.libappbase.utils;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * @Author ZhanGSKen<zhangsken@188.com>
 | 
				
			||||||
 | 
					 * @Date 2025/06/04 20:15
 | 
				
			||||||
 | 
					 * @Describe 文件操作类
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					import java.io.BufferedReader;
 | 
				
			||||||
 | 
					import java.io.BufferedWriter;
 | 
				
			||||||
 | 
					import java.io.ByteArrayOutputStream;
 | 
				
			||||||
 | 
					import java.io.FileInputStream;
 | 
				
			||||||
 | 
					import java.io.FileOutputStream;
 | 
				
			||||||
 | 
					import java.io.FileReader;
 | 
				
			||||||
 | 
					import java.io.FileWriter;
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class FileUtils {
 | 
				
			||||||
 | 
					    public static final String TAG = "FileUtils";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * 读取文件为字节数组(Java 7 语法)
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    public static byte[] readFileToByteArray(String filePath) {
 | 
				
			||||||
 | 
					        FileInputStream fis = null;
 | 
				
			||||||
 | 
					        ByteArrayOutputStream bos = null;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            fis = new FileInputStream(filePath);
 | 
				
			||||||
 | 
					            bos = new ByteArrayOutputStream();
 | 
				
			||||||
 | 
					            byte[] buffer = new byte[4096];
 | 
				
			||||||
 | 
					            int bytesRead;
 | 
				
			||||||
 | 
					            while ((bytesRead = fis.read(buffer)) != -1) {
 | 
				
			||||||
 | 
					                bos.write(buffer, 0, bytesRead);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return bos.toByteArray();
 | 
				
			||||||
 | 
					        } catch (IOException e) {
 | 
				
			||||||
 | 
					            e.printStackTrace();
 | 
				
			||||||
 | 
					            return null;
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            // 手动关闭流(Java 7 不支持 try-with-resources)
 | 
				
			||||||
 | 
					            if (fis != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    fis.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    e.printStackTrace();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            if (bos != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    bos.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    e.printStackTrace();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * 写入字节数组到文件(Java 7 语法)
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    public static boolean writeByteArrayToFile(byte[] data, String filePath) {
 | 
				
			||||||
 | 
					        FileOutputStream fos = null;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            fos = new FileOutputStream(filePath);
 | 
				
			||||||
 | 
					            fos.write(data);
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        } catch (IOException e) {
 | 
				
			||||||
 | 
					            e.printStackTrace();
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            if (fos != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    fos.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    e.printStackTrace();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // 原字符串读写方法(适配 Java 7)
 | 
				
			||||||
 | 
					    public static String readFileToString(String filePath) {
 | 
				
			||||||
 | 
					        BufferedReader reader = null;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            reader = new BufferedReader(new FileReader(filePath));
 | 
				
			||||||
 | 
					            StringBuilder content = new StringBuilder();
 | 
				
			||||||
 | 
					            String line;
 | 
				
			||||||
 | 
					            while ((line = reader.readLine()) != null) {
 | 
				
			||||||
 | 
					                content.append(line).append(System.getProperty("line.separator"));
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            // 去除最后一个换行符(可选)
 | 
				
			||||||
 | 
					            if (content.length() > 0) {
 | 
				
			||||||
 | 
					                content.deleteCharAt(content.length() - 1);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return content.toString();
 | 
				
			||||||
 | 
					        } catch (IOException e) {
 | 
				
			||||||
 | 
					            e.printStackTrace();
 | 
				
			||||||
 | 
					            return null;
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            if (reader != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    reader.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    e.printStackTrace();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static boolean writeStringToFile(String content, String filePath, boolean append) {
 | 
				
			||||||
 | 
					        BufferedWriter writer = null;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            writer = new BufferedWriter(new FileWriter(filePath, append));
 | 
				
			||||||
 | 
					            writer.write(content);
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        } catch (IOException e) {
 | 
				
			||||||
 | 
					            e.printStackTrace();
 | 
				
			||||||
 | 
					            return false;
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            if (writer != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    writer.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    e.printStackTrace();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -5,81 +5,98 @@ package cc.winboll.studio.libappbase.utils;
 | 
				
			|||||||
 * @Date 2025/06/04 13:36
 | 
					 * @Date 2025/06/04 13:36
 | 
				
			||||||
 * @Describe RSA加密工具
 | 
					 * @Describe RSA加密工具
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					import android.content.Context;
 | 
				
			||||||
 | 
					import android.util.Base64;
 | 
				
			||||||
import cc.winboll.studio.libappbase.LogUtils;
 | 
					import cc.winboll.studio.libappbase.LogUtils;
 | 
				
			||||||
import java.io.File;
 | 
					import java.io.File;
 | 
				
			||||||
 | 
					import java.io.FileInputStream;
 | 
				
			||||||
import java.io.FileOutputStream;
 | 
					import java.io.FileOutputStream;
 | 
				
			||||||
import java.io.IOException;
 | 
					import java.io.IOException;
 | 
				
			||||||
import java.nio.file.Files;
 | 
					 | 
				
			||||||
import java.nio.file.Paths;
 | 
					 | 
				
			||||||
import java.security.KeyFactory;
 | 
					import java.security.KeyFactory;
 | 
				
			||||||
import java.security.KeyPair;
 | 
					import java.security.KeyPair;
 | 
				
			||||||
import java.security.KeyPairGenerator;
 | 
					import java.security.KeyPairGenerator;
 | 
				
			||||||
 | 
					import java.security.NoSuchAlgorithmException;
 | 
				
			||||||
import java.security.PrivateKey;
 | 
					import java.security.PrivateKey;
 | 
				
			||||||
import java.security.PublicKey;
 | 
					import java.security.PublicKey;
 | 
				
			||||||
 | 
					import java.security.spec.InvalidKeySpecException;
 | 
				
			||||||
import java.security.spec.PKCS8EncodedKeySpec;
 | 
					import java.security.spec.PKCS8EncodedKeySpec;
 | 
				
			||||||
import java.security.spec.X509EncodedKeySpec;
 | 
					import java.security.spec.X509EncodedKeySpec;
 | 
				
			||||||
 | 
					import java.util.Objects;
 | 
				
			||||||
import javax.crypto.Cipher;
 | 
					import javax.crypto.Cipher;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
public class RSAUtils {
 | 
					public class RSAUtils {
 | 
				
			||||||
    public static final String TAG = "RSAUtils";
 | 
					    private static final String TAG = "RSAUtils";
 | 
				
			||||||
    private static final RSAUtils INSTANCE = new RSAUtils();
 | 
					 | 
				
			||||||
    private static final int KEY_SIZE = 2048;
 | 
					    private static final int KEY_SIZE = 2048;
 | 
				
			||||||
    private static final String KEY_ALGORITHM = "RSA";
 | 
					    private static final String KEY_ALGORITHM = "RSA";
 | 
				
			||||||
    private static final String PUBLIC_KEY_FILE = "public.key";
 | 
					    private static final String PUBLIC_KEY_FILE = "public.key";
 | 
				
			||||||
    private static final String PRIVATE_KEY_FILE = "private.key";
 | 
					    private static final String PRIVATE_KEY_FILE = "private.key";
 | 
				
			||||||
 | 
					    private static final String CIPHER_ALGORITHM = KEY_ALGORITHM + "/ECB/PKCS1Padding"; // 保留原加密方式
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    private RSAUtils() {}
 | 
					    private final String keyPath;
 | 
				
			||||||
 | 
					    private static volatile RSAUtils INSTANCE;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public static RSAUtils getInstance() {
 | 
					    /**
 | 
				
			||||||
 | 
					     * 构造方法:初始化密钥存储路径(内部存储)
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    private RSAUtils(Context context) {
 | 
				
			||||||
 | 
					        keyPath = context.getFilesDir() + File.separator + "keys" + File.separator; // 修正路径格式
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * 获取单例实例
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    public static synchronized RSAUtils getInstance(Context context) {
 | 
				
			||||||
 | 
					        if (INSTANCE == null) {
 | 
				
			||||||
 | 
					            INSTANCE = new RSAUtils(context);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        return INSTANCE;
 | 
					        return INSTANCE;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 检查密钥文件是否存在
 | 
					     * 检查密钥文件是否存在
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public boolean keysExist(String path) {
 | 
					    public boolean keysExist() {
 | 
				
			||||||
        File publicKeyFile = new File(path + PUBLIC_KEY_FILE);
 | 
					        File publicKeyFile = new File(keyPath + PUBLIC_KEY_FILE);
 | 
				
			||||||
        File privateKeyFile = new File(path + PRIVATE_KEY_FILE);
 | 
					        File privateKeyFile = new File(keyPath + PRIVATE_KEY_FILE);
 | 
				
			||||||
        return publicKeyFile.exists() && privateKeyFile.exists();
 | 
					        return publicKeyFile.exists() && privateKeyFile.exists();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 生成密钥对并保存为字节文件
 | 
					     * 生成密钥对并保存到文件
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public void generateAndSaveKeys(String path) throws Exception {
 | 
					    public void generateAndSaveKeys() throws Exception {
 | 
				
			||||||
        LogUtils.d(TAG, "正在生成密钥对");
 | 
					        LogUtils.d(TAG, "开始生成 RSA 密钥对(2048位)");
 | 
				
			||||||
        // 生成密钥对
 | 
					 | 
				
			||||||
        KeyPairGenerator generator = KeyPairGenerator.getInstance(KEY_ALGORITHM);
 | 
					        KeyPairGenerator generator = KeyPairGenerator.getInstance(KEY_ALGORITHM);
 | 
				
			||||||
        generator.initialize(KEY_SIZE);
 | 
					        generator.initialize(KEY_SIZE);
 | 
				
			||||||
        KeyPair keyPair = generator.generateKeyPair();
 | 
					        KeyPair keyPair = generator.generateKeyPair();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // 保存公钥(X.509字节)
 | 
					        saveKey(PUBLIC_KEY_FILE, keyPair.getPublic().getEncoded());
 | 
				
			||||||
        saveKey(path, PUBLIC_KEY_FILE, keyPair.getPublic().getEncoded());
 | 
					        saveKey(PRIVATE_KEY_FILE, keyPair.getPrivate().getEncoded());
 | 
				
			||||||
        // 保存私钥(PKCS#8字节)
 | 
					        LogUtils.d(TAG, "密钥对生成并保存成功");
 | 
				
			||||||
        saveKey(path, PRIVATE_KEY_FILE, keyPair.getPrivate().getEncoded());
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 读取密钥对(存在则读取,不存在则生成)
 | 
					     * 获取或生成密钥对(线程安全)
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public KeyPair getOrGenerateKeys(String path) throws Exception {
 | 
					    public KeyPair getOrGenerateKeys() throws Exception {
 | 
				
			||||||
        if (keysExist(path)) {
 | 
					        if (!keysExist()) {
 | 
				
			||||||
            LogUtils.d(TAG, "密钥对已存在");
 | 
					            synchronized (RSAUtils.class) { // 双重检查锁,避免多线程重复生成
 | 
				
			||||||
            return readKeysFromFile(path);
 | 
					                if (!keysExist()) {
 | 
				
			||||||
        } else {
 | 
					                    generateAndSaveKeys();
 | 
				
			||||||
            LogUtils.d(TAG, "未生成密钥对");
 | 
					 | 
				
			||||||
            generateAndSaveKeys(path);
 | 
					 | 
				
			||||||
            return readKeysFromFile(path);
 | 
					 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return readKeysFromFile();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 从文件读取密钥对
 | 
					     * 从文件读取密钥对
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    private KeyPair readKeysFromFile(String path) throws Exception {
 | 
					    private KeyPair readKeysFromFile() throws Exception {
 | 
				
			||||||
        LogUtils.d(TAG, "正在读取密钥对");
 | 
					        LogUtils.d(TAG, "读取密钥对文件");
 | 
				
			||||||
        byte[] publicKeyBytes = Files.readAllBytes(Paths.get(path + PUBLIC_KEY_FILE));
 | 
					        try {
 | 
				
			||||||
        byte[] privateKeyBytes = Files.readAllBytes(Paths.get(path + PRIVATE_KEY_FILE));
 | 
					            byte[] publicKeyBytes = readFileToBytes(keyPath + PUBLIC_KEY_FILE);
 | 
				
			||||||
 | 
					            byte[] privateKeyBytes = readFileToBytes(keyPath + PRIVATE_KEY_FILE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            X509EncodedKeySpec publicSpec = new X509EncodedKeySpec(publicKeyBytes);
 | 
					            X509EncodedKeySpec publicSpec = new X509EncodedKeySpec(publicKeyBytes);
 | 
				
			||||||
            PKCS8EncodedKeySpec privateSpec = new PKCS8EncodedKeySpec(privateKeyBytes);
 | 
					            PKCS8EncodedKeySpec privateSpec = new PKCS8EncodedKeySpec(privateKeyBytes);
 | 
				
			||||||
@@ -89,36 +106,117 @@ public class RSAUtils {
 | 
				
			|||||||
            PrivateKey privateKey = factory.generatePrivate(privateSpec);
 | 
					            PrivateKey privateKey = factory.generatePrivate(privateSpec);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            return new KeyPair(publicKey, privateKey);
 | 
					            return new KeyPair(publicKey, privateKey);
 | 
				
			||||||
 | 
					        } catch (IOException | NoSuchAlgorithmException | InvalidKeySpecException e) {
 | 
				
			||||||
 | 
					            LogUtils.e(TAG, "密钥文件读取失败:" + e.getMessage());
 | 
				
			||||||
 | 
					            throw new Exception("密钥文件损坏或格式错误", e);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 通用保存密钥字节方法
 | 
					     * 保存密钥到文件(通用方法)
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    private void saveKey(String path, String fileName, byte[] keyBytes) throws IOException {
 | 
					    private void saveKey(String fileName, byte[] keyBytes) throws IOException {
 | 
				
			||||||
        File dir = new File(path);
 | 
					        Objects.requireNonNull(keyBytes, "密钥字节数据不可为空");
 | 
				
			||||||
        if (!dir.exists()) dir.mkdirs(); // 自动创建目录
 | 
					        File dir = new File(keyPath);
 | 
				
			||||||
        try (FileOutputStream fos = new FileOutputStream(path + fileName)) {
 | 
					        if (!dir.exists() && !dir.mkdirs()) {
 | 
				
			||||||
 | 
					            throw new IOException("创建密钥目录失败:" + keyPath);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        FileOutputStream fos = null;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            fos = new FileOutputStream(keyPath + fileName);
 | 
				
			||||||
            fos.write(keyBytes);
 | 
					            fos.write(keyBytes);
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            if (fos != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    fos.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    LogUtils.e(TAG, "关闭文件流失败:" + e.getMessage());
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 公钥加密
 | 
					     * 读取文件为字节数组(Java 7 兼容)
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    private byte[] readFileToBytes(String filePath) throws IOException {
 | 
				
			||||||
 | 
					        File file = new File(filePath);
 | 
				
			||||||
 | 
					        if (!file.exists() || file.isDirectory()) {
 | 
				
			||||||
 | 
					            throw new IOException("文件不存在或为目录:" + filePath);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        FileInputStream fis = null;
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            fis = new FileInputStream(file);
 | 
				
			||||||
 | 
					            byte[] data = new byte[(int) file.length()];
 | 
				
			||||||
 | 
					            int bytesRead = fis.read(data);
 | 
				
			||||||
 | 
					            if (bytesRead != data.length) {
 | 
				
			||||||
 | 
					                throw new IOException("文件读取不完整");
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return data;
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            if (fis != null) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    fis.close();
 | 
				
			||||||
 | 
					                } catch (IOException e) {
 | 
				
			||||||
 | 
					                    LogUtils.e(TAG, "关闭文件流失败:" + e.getMessage());
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * 公钥加密(带参数校验)
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public byte[] encryptWithPublicKey(String plainText, PublicKey publicKey) throws Exception {
 | 
					    public byte[] encryptWithPublicKey(String plainText, PublicKey publicKey) throws Exception {
 | 
				
			||||||
        Cipher cipher = Cipher.getInstance(KEY_ALGORITHM + "/ECB/PKCS1Padding");
 | 
					        Objects.requireNonNull(plainText, "明文不可为空");
 | 
				
			||||||
 | 
					        Objects.requireNonNull(publicKey, "公钥不可为空");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Cipher cipher = Cipher.getInstance(CIPHER_ALGORITHM);
 | 
				
			||||||
        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
 | 
					        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 检查数据长度是否超过 RSA 限制(2048位密钥最大明文为 214字节,PKCS1Padding)
 | 
				
			||||||
 | 
					        int maxPlainTextSize = cipher.getBlockSize() - 11; // PKCS1Padding 固定填充长度
 | 
				
			||||||
 | 
					        if (plainText.getBytes("UTF-8").length > maxPlainTextSize) {
 | 
				
			||||||
 | 
					            throw new IllegalArgumentException("明文过长,最大支持 " + maxPlainTextSize + " 字节");
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return cipher.doFinal(plainText.getBytes("UTF-8"));
 | 
					        return cipher.doFinal(plainText.getBytes("UTF-8"));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * 私钥解密
 | 
					     * 私钥解密(带参数校验)
 | 
				
			||||||
     */
 | 
					     */
 | 
				
			||||||
    public String decryptWithPrivateKey(byte[] encryptedData, PrivateKey privateKey) throws Exception {
 | 
					    public String decryptWithPrivateKey(byte[] encryptedData, PrivateKey privateKey) throws Exception {
 | 
				
			||||||
        Cipher cipher = Cipher.getInstance(KEY_ALGORITHM + "/ECB/PKCS1Padding");
 | 
					        Objects.requireNonNull(encryptedData, "密文不可为空");
 | 
				
			||||||
 | 
					        Objects.requireNonNull(privateKey, "私钥不可为空");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Cipher cipher = Cipher.getInstance(CIPHER_ALGORITHM);
 | 
				
			||||||
        cipher.init(Cipher.DECRYPT_MODE, privateKey);
 | 
					        cipher.init(Cipher.DECRYPT_MODE, privateKey);
 | 
				
			||||||
        byte[] decryptedBytes = cipher.doFinal(encryptedData);
 | 
					        byte[] decryptedBytes = cipher.doFinal(encryptedData);
 | 
				
			||||||
        return new String(decryptedBytes, "UTF-8");
 | 
					        return new String(decryptedBytes, "UTF-8");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * 将 HTTP 传输的 Base64 字符串还原为加密字节数组(Java 7 兼容)
 | 
				
			||||||
 | 
					     * @param httpString Base64 字符串(非 null)
 | 
				
			||||||
 | 
					     * @return 加密字节数组
 | 
				
			||||||
 | 
					     * @throws IllegalArgumentException 解码失败时抛出
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    public byte[] httpStringToEncryptBytes(String httpString) {
 | 
				
			||||||
 | 
					        Objects.requireNonNull(httpString, "HTTP 字符串不可为空");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 计算缺失的填充符数量(Java 7 不支持 repeat(),手动拼接)
 | 
				
			||||||
 | 
					        int pad = httpString.length() % 4;
 | 
				
			||||||
 | 
					        StringBuilder paddedString = new StringBuilder(httpString);
 | 
				
			||||||
 | 
					        if (pad != 0) {
 | 
				
			||||||
 | 
					            for (int i = 0; i < pad; i++) {
 | 
				
			||||||
 | 
					                paddedString.append('='); // 补全 '='
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 使用 Base64 解码(Android 原生 Base64 类兼容 Java 7)
 | 
				
			||||||
 | 
					        return Base64.decode(paddedString.toString(), Base64.URL_SAFE);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1,281 @@
 | 
				
			|||||||
 | 
					package cc.winboll.studio.libappbase.utils;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * @Author ZhanGSKen<zhangsken@188.com>
 | 
				
			||||||
 | 
					 * @Date 2025/06/04 17:21
 | 
				
			||||||
 | 
					 * @Describe 应用登录与接口工具
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					import android.content.Context;
 | 
				
			||||||
 | 
					import android.os.Handler;
 | 
				
			||||||
 | 
					import android.os.Looper;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.LogUtils;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.models.ResponseData;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.models.UserInfoModel;
 | 
				
			||||||
 | 
					import com.google.gson.Gson;
 | 
				
			||||||
 | 
					import java.io.File;
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					import java.net.URLDecoder;
 | 
				
			||||||
 | 
					import java.nio.charset.StandardCharsets;
 | 
				
			||||||
 | 
					import java.security.KeyPair;
 | 
				
			||||||
 | 
					import java.security.PrivateKey;
 | 
				
			||||||
 | 
					import java.security.PublicKey;
 | 
				
			||||||
 | 
					import java.util.concurrent.TimeUnit;
 | 
				
			||||||
 | 
					import okhttp3.Call;
 | 
				
			||||||
 | 
					import okhttp3.Callback;
 | 
				
			||||||
 | 
					import okhttp3.MediaType;
 | 
				
			||||||
 | 
					import okhttp3.OkHttpClient;
 | 
				
			||||||
 | 
					import okhttp3.Request;
 | 
				
			||||||
 | 
					import okhttp3.RequestBody;
 | 
				
			||||||
 | 
					import okhttp3.Response;
 | 
				
			||||||
 | 
					import java.io.UnsupportedEncodingException;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class YunUtils {
 | 
				
			||||||
 | 
					    public static final String TAG = "YunUtils";
 | 
				
			||||||
 | 
					    // 私有静态实例,类加载时创建
 | 
				
			||||||
 | 
					    private static volatile YunUtils INSTANCE;
 | 
				
			||||||
 | 
					    Context mContext;
 | 
				
			||||||
 | 
					    UserInfoModel mUserInfoModel;
 | 
				
			||||||
 | 
					    String token = "";
 | 
				
			||||||
 | 
					    String mDataFolderPath = "";
 | 
				
			||||||
 | 
					    String mUserInfoModelPath = "";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static final int CONNECT_TIMEOUT = 15; // 连接超时时间(秒)
 | 
				
			||||||
 | 
					    private static final int READ_TIMEOUT = 20;    // 读取超时时间(秒)
 | 
				
			||||||
 | 
					    private static volatile YunUtils instance;
 | 
				
			||||||
 | 
					    private OkHttpClient okHttpClient;
 | 
				
			||||||
 | 
					    private Handler mainHandler; // 主线程 Handler
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // 私有构造方法,防止外部实例化
 | 
				
			||||||
 | 
					    private YunUtils(Context context) {
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "YunUtils");
 | 
				
			||||||
 | 
					        mContext = context;
 | 
				
			||||||
 | 
					        mDataFolderPath = mContext.getExternalFilesDir(TAG).toString();
 | 
				
			||||||
 | 
					        File fTest = new File(mDataFolderPath);
 | 
				
			||||||
 | 
					        if (!fTest.exists()) {
 | 
				
			||||||
 | 
					            fTest.mkdirs();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        mUserInfoModelPath = mDataFolderPath + File.separator + "UserInfoModel.rsajson";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        okHttpClient = new OkHttpClient.Builder()
 | 
				
			||||||
 | 
					            .connectTimeout(CONNECT_TIMEOUT, TimeUnit.SECONDS)
 | 
				
			||||||
 | 
					            .readTimeout(READ_TIMEOUT, TimeUnit.SECONDS)
 | 
				
			||||||
 | 
					            .build();
 | 
				
			||||||
 | 
					        mainHandler = new Handler(Looper.getMainLooper()); // 获取主线程 Looper
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // 公共静态方法,返回唯一实例
 | 
				
			||||||
 | 
					    public static synchronized YunUtils getInstance(Context context) {
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "getInstance");
 | 
				
			||||||
 | 
					        if (INSTANCE == null) {
 | 
				
			||||||
 | 
					            INSTANCE = new YunUtils(context);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return INSTANCE;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void checkLoginStatus() {
 | 
				
			||||||
 | 
					        String token = getLocalToken();
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, String.format("checkLoginStatus token is %s", token));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    String getLocalToken() {
 | 
				
			||||||
 | 
					        UserInfoModel userInfoModel = loadUserInfoModel();
 | 
				
			||||||
 | 
					        return (userInfoModel == null) ?"": userInfoModel.getToken();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void login(String host, UserInfoModel userInfoModel) {
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "login");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // 发送 POST 请求
 | 
				
			||||||
 | 
					        String apiUrl = host + "/login/index.php";
 | 
				
			||||||
 | 
					        // 序列化对象为JSON
 | 
				
			||||||
 | 
					        Gson gson = new Gson();
 | 
				
			||||||
 | 
					        String jsonData = gson.toJson(userInfoModel); // 自动生成标准JSON
 | 
				
			||||||
 | 
					        //String jsonData = userInfoModel.toString();
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "要发送的数据 : " + jsonData);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        sendPostRequest(apiUrl, jsonData, new OnResponseListener() {
 | 
				
			||||||
 | 
					                // 成功回调(主线程)
 | 
				
			||||||
 | 
					                @Override
 | 
				
			||||||
 | 
					                public void onSuccess(String responseBody) {
 | 
				
			||||||
 | 
					                    LogUtils.d(TAG, "onSuccess");
 | 
				
			||||||
 | 
					                    LogUtils.d(TAG, String.format("responseBody %s", responseBody));
 | 
				
			||||||
 | 
					                    Gson gson = new Gson();
 | 
				
			||||||
 | 
					                    ResponseData result = gson.fromJson(responseBody, ResponseData.class); // 转为 Result 实例
 | 
				
			||||||
 | 
					                    if(result.getStatus().equals(ResponseData.STATUS_SUCCESS)) {
 | 
				
			||||||
 | 
					                        
 | 
				
			||||||
 | 
					                            UserInfoModel userInfoModel = result.getData();
 | 
				
			||||||
 | 
					                            if (userInfoModel != null) {
 | 
				
			||||||
 | 
					                                LogUtils.d(TAG, "收到网站 UserInfoModel");
 | 
				
			||||||
 | 
					                                String token = userInfoModel.getToken();
 | 
				
			||||||
 | 
					                                saveLocalToken(token);
 | 
				
			||||||
 | 
					                                checkLoginStatus();
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                       
 | 
				
			||||||
 | 
					                    } else if(result.getStatus().equals(ResponseData.STATUS_ERROR)) {
 | 
				
			||||||
 | 
					                        try {
 | 
				
			||||||
 | 
					                            String decodedMessage = URLDecoder.decode(result.getMessage(), "UTF-8");
 | 
				
			||||||
 | 
					                            LogUtils.d(TAG, "服务器返回信息: " + decodedMessage);
 | 
				
			||||||
 | 
					                        } catch (UnsupportedEncodingException e) {
 | 
				
			||||||
 | 
					                            LogUtils.d(TAG, e, Thread.currentThread().getStackTrace());
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                // 失败回调(主线程)
 | 
				
			||||||
 | 
					                @Override
 | 
				
			||||||
 | 
					                public void onFailure(String errorMsg) {
 | 
				
			||||||
 | 
					                    LogUtils.d(TAG, errorMsg);
 | 
				
			||||||
 | 
					                    // 处理错误
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public void saveLocalToken(String token) {
 | 
				
			||||||
 | 
					        UserInfoModel userInfoModel = new UserInfoModel();
 | 
				
			||||||
 | 
					        userInfoModel.setToken(token);
 | 
				
			||||||
 | 
					        saveUserInfoModel(userInfoModel);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    UserInfoModel loadUserInfoModel() {
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "loadUserInfoModel");
 | 
				
			||||||
 | 
					        if (new File(mUserInfoModelPath).exists()) {
 | 
				
			||||||
 | 
					            try {
 | 
				
			||||||
 | 
					                // 加载加密后的模型数据
 | 
				
			||||||
 | 
					                byte[] encryptedData = FileUtils.readFileToByteArray(mUserInfoModelPath);
 | 
				
			||||||
 | 
					                // 加载 RSA 工具
 | 
				
			||||||
 | 
					                RSAUtils utils = RSAUtils.getInstance(mContext);
 | 
				
			||||||
 | 
					                KeyPair keyPair = utils.getOrGenerateKeys();
 | 
				
			||||||
 | 
					                //PublicKey publicKey = keyPair.getPublic();
 | 
				
			||||||
 | 
					                PrivateKey privateKey = keyPair.getPrivate();
 | 
				
			||||||
 | 
					                // 私钥解密模型数据
 | 
				
			||||||
 | 
					                String szInfo = utils.decryptWithPrivateKey(encryptedData, keyPair.getPrivate());
 | 
				
			||||||
 | 
					                LogUtils.d(TAG, String.format("szInfo %s", szInfo));
 | 
				
			||||||
 | 
					                mUserInfoModel = UserInfoModel.parseStringToBean(szInfo, UserInfoModel.class);
 | 
				
			||||||
 | 
					                if (mUserInfoModel == null) {
 | 
				
			||||||
 | 
					                    LogUtils.d(TAG, "模型数据解析为空数据。");
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                LogUtils.d(TAG, "UserInfoModel 解密加载结束。");
 | 
				
			||||||
 | 
					            } catch (Exception e) {
 | 
				
			||||||
 | 
					                LogUtils.d(TAG, e, Thread.currentThread().getStackTrace());
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            LogUtils.d(TAG, "云服务登录信息不存在。");
 | 
				
			||||||
 | 
					            mUserInfoModel = null;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return mUserInfoModel;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void saveUserInfoModel(UserInfoModel userInfoModel) {
 | 
				
			||||||
 | 
					        LogUtils.d(TAG, "saveUserInfoModel");
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            String szInfo = userInfoModel.toString();
 | 
				
			||||||
 | 
					            LogUtils.d(TAG, "原始数据: " + szInfo);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            RSAUtils utils = RSAUtils.getInstance(mContext);
 | 
				
			||||||
 | 
					            KeyPair keyPair = utils.getOrGenerateKeys();
 | 
				
			||||||
 | 
					            PublicKey publicKey = keyPair.getPublic();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            // 公钥加密(传入字节数组,避免中间字符串转换)
 | 
				
			||||||
 | 
					            byte[] encryptedData = utils.encryptWithPublicKey(szInfo, publicKey);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            // 保存加密字节数组到文件(直接操作字节,无需转字符串)
 | 
				
			||||||
 | 
					            FileUtils.writeByteArrayToFile(encryptedData, mUserInfoModelPath);
 | 
				
			||||||
 | 
					            LogUtils.d(TAG, "加密数据已保存");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            // 测试解密(仅调试用)
 | 
				
			||||||
 | 
					            String szInfo2 = utils.decryptWithPrivateKey(encryptedData, keyPair.getPrivate());
 | 
				
			||||||
 | 
					            LogUtils.d(TAG, "解密结果: " + szInfo2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            mUserInfoModel = UserInfoModel.parseStringToBean(szInfo2, UserInfoModel.class);
 | 
				
			||||||
 | 
					            if (mUserInfoModel == null) {
 | 
				
			||||||
 | 
					                LogUtils.d(TAG, "模型解析失败");
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        } catch (Exception e) {
 | 
				
			||||||
 | 
					            LogUtils.d(TAG, "加密/解密失败: " + e.getMessage());
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // 发送 POST 请求(JSON 数据)
 | 
				
			||||||
 | 
					    public void sendPostRequest(String url, String data, OnResponseListener listener) {
 | 
				
			||||||
 | 
					        RequestBody requestBody = RequestBody.create(
 | 
				
			||||||
 | 
					            MediaType.parse("application/json; charset=utf-8"), // 关键头信息
 | 
				
			||||||
 | 
					            data.getBytes(StandardCharsets.UTF_8)
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Request request = new Request.Builder()
 | 
				
			||||||
 | 
					            .url(url)
 | 
				
			||||||
 | 
					            .post(requestBody)
 | 
				
			||||||
 | 
					            .addHeader("Content-Type", "application/json") // 显式添加头
 | 
				
			||||||
 | 
					            .build();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        executeRequest(request, listener);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // 发送 GET 请求
 | 
				
			||||||
 | 
					    public void sendGetRequest(String url, OnResponseListener listener) {
 | 
				
			||||||
 | 
					        Request request = new Request.Builder()
 | 
				
			||||||
 | 
					            .url(url)
 | 
				
			||||||
 | 
					            .get()
 | 
				
			||||||
 | 
					            .build();
 | 
				
			||||||
 | 
					        executeRequest(request, listener);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // 执行请求(子线程处理)
 | 
				
			||||||
 | 
					    private void executeRequest(final Request request, final OnResponseListener listener) {
 | 
				
			||||||
 | 
					        okHttpClient.newCall(request).enqueue(new Callback() {
 | 
				
			||||||
 | 
					                // 响应成功(子线程)
 | 
				
			||||||
 | 
					                @Override
 | 
				
			||||||
 | 
					                public void onResponse(Call call, Response response) throws IOException {
 | 
				
			||||||
 | 
					                    try {
 | 
				
			||||||
 | 
					                        if (!response.isSuccessful()) {
 | 
				
			||||||
 | 
					                            postFailure(listener, "响应码错误:" + response.code());
 | 
				
			||||||
 | 
					                            return;
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                        String responseBody = response.body().string();
 | 
				
			||||||
 | 
					                        postSuccess(listener, responseBody);
 | 
				
			||||||
 | 
					                    } catch (Exception e) {
 | 
				
			||||||
 | 
					                        postFailure(listener, "解析失败:" + e.getMessage());
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                // 响应失败(子线程)
 | 
				
			||||||
 | 
					                @Override
 | 
				
			||||||
 | 
					                public void onFailure(Call call, IOException e) {
 | 
				
			||||||
 | 
					                    postFailure(listener, "网络失败:" + e.getMessage());
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                // 主线程回调(使用 Handler)
 | 
				
			||||||
 | 
					                private void postSuccess(final OnResponseListener listener, final String msg) {
 | 
				
			||||||
 | 
					                    mainHandler.post(new Runnable() {
 | 
				
			||||||
 | 
					                            @Override
 | 
				
			||||||
 | 
					                            public void run() {
 | 
				
			||||||
 | 
					                                listener.onSuccess(msg);
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                private void postFailure(final OnResponseListener listener, final String msg) {
 | 
				
			||||||
 | 
					                    mainHandler.post(new Runnable() {
 | 
				
			||||||
 | 
					                            @Override
 | 
				
			||||||
 | 
					                            public void run() {
 | 
				
			||||||
 | 
					                                listener.onFailure(msg);
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public interface OnResponseListener {
 | 
				
			||||||
 | 
					        /**
 | 
				
			||||||
 | 
					         * 成功响应(主线程回调)
 | 
				
			||||||
 | 
					         * @param responseBody 响应体字符串
 | 
				
			||||||
 | 
					         */
 | 
				
			||||||
 | 
					        void onSuccess(String responseBody);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /**
 | 
				
			||||||
 | 
					         * 失败回调(包含错误信息)
 | 
				
			||||||
 | 
					         * @param errorMsg 错误描述
 | 
				
			||||||
 | 
					         */
 | 
				
			||||||
 | 
					        void onFailure(String errorMsg);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -6,6 +6,31 @@
 | 
				
			|||||||
	android:layout_width="match_parent"
 | 
						android:layout_width="match_parent"
 | 
				
			||||||
	android:layout_height="match_parent">
 | 
						android:layout_height="match_parent">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <LinearLayout
 | 
				
			||||||
 | 
					        android:orientation="horizontal"
 | 
				
			||||||
 | 
					        android:layout_width="match_parent"
 | 
				
			||||||
 | 
					        android:layout_height="wrap_content"
 | 
				
			||||||
 | 
					        android:layout_gravity="right"
 | 
				
			||||||
 | 
					        android:gravity="right"
 | 
				
			||||||
 | 
					        android:padding="10dp"
 | 
				
			||||||
 | 
					        android:id="@+id/ll_hostbar">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <RadioButton
 | 
				
			||||||
 | 
					            android:layout_width="wrap_content"
 | 
				
			||||||
 | 
					            android:layout_height="wrap_content"
 | 
				
			||||||
 | 
					            android:text="10.8.0.250:456"
 | 
				
			||||||
 | 
					            android:id="@+id/rb_debughost"
 | 
				
			||||||
 | 
					            android:onClick="onSwitchHost"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <RadioButton
 | 
				
			||||||
 | 
					            android:layout_width="wrap_content"
 | 
				
			||||||
 | 
					            android:layout_height="wrap_content"
 | 
				
			||||||
 | 
					            android:text="yun.winboll.cc"
 | 
				
			||||||
 | 
					            android:id="@+id/rb_yunhost"
 | 
				
			||||||
 | 
					            android:onClick="onSwitchHost"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						</LinearLayout>
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
	<LinearLayout
 | 
						<LinearLayout
 | 
				
			||||||
		android:orientation="horizontal"
 | 
							android:orientation="horizontal"
 | 
				
			||||||
		android:layout_width="match_parent"
 | 
							android:layout_width="match_parent"
 | 
				
			||||||
@@ -18,6 +43,12 @@
 | 
				
			|||||||
			android:text="Test RSA"
 | 
								android:text="Test RSA"
 | 
				
			||||||
            android:onClick="onTestRSA"/>
 | 
					            android:onClick="onTestRSA"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <Button
 | 
				
			||||||
 | 
					            android:layout_width="wrap_content"
 | 
				
			||||||
 | 
					            android:layout_height="wrap_content"
 | 
				
			||||||
 | 
					            android:text="Test Login"
 | 
				
			||||||
 | 
					            android:onClick="onTestLogin"/>
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
	</LinearLayout>
 | 
						</LinearLayout>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	<LinearLayout
 | 
						<LinearLayout
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										1
									
								
								numtable/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					/build
 | 
				
			||||||
							
								
								
									
										34
									
								
								numtable/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,34 @@
 | 
				
			|||||||
 | 
					# NumTable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### 介绍
 | 
				
			||||||
 | 
					桌面图标多元应用。提供一个数字风格化的桌面标识图标,快捷的桌面标识创建途径。主要应用于桌面繁多时的页面环境辅助识别。
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### 软件架构
 | 
				
			||||||
 | 
					适配安卓应用 [AIDE Pro] 的 Gradle 编译结构。
 | 
				
			||||||
 | 
					也适配安卓应用 [AndroidIDE] 的 Gradle 编译结构。
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### Gradle 编译说明
 | 
				
			||||||
 | 
					调试版编译命令 :gradle assembleBetaDebug
 | 
				
			||||||
 | 
					阶段版编译命令 :bash .winboll/bashPublishAPKAddTag.sh numtable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### 使用说明
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### 参与贡献
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					1.  Fork 本仓库
 | 
				
			||||||
 | 
					2.  新建 Feat_xxx 分支
 | 
				
			||||||
 | 
					3.  提交代码 : ZhanGSKen(ZhanGSKen<zhangsken@188.com>)
 | 
				
			||||||
 | 
					4.  新建 Pull Request
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### 特技
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					1.  使用 Readme\_XXX.md 来支持不同的语言,例如 Readme\_en.md, Readme\_zh.md
 | 
				
			||||||
 | 
					2.  Gitee 官方博客 [blog.gitee.com](https://blog.gitee.com)
 | 
				
			||||||
 | 
					3.  你可以 [https://gitee.com/explore](https://gitee.com/explore) 这个地址来了解 Gitee 上的优秀开源项目
 | 
				
			||||||
 | 
					4.  [GVP](https://gitee.com/gvp) 全称是 Gitee 最有价值开源项目,是综合评定出的优秀开源项目
 | 
				
			||||||
 | 
					5.  Gitee 官方提供的使用手册 [https://gitee.com/help](https://gitee.com/help)
 | 
				
			||||||
 | 
					6.  Gitee 封面人物是一档用来展示 Gitee 会员风采的栏目 [https://gitee.com/gitee-stars/](https://gitee.com/gitee-stars/)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#### 参考文档
 | 
				
			||||||
							
								
								
									
										1
									
								
								numtable/app_update_description.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										73
									
								
								numtable/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,73 @@
 | 
				
			|||||||
 | 
					apply plugin: 'com.android.application'
 | 
				
			||||||
 | 
					apply from: '../.winboll/winboll_app_build.gradle'
 | 
				
			||||||
 | 
					apply from: '../.winboll/winboll_lint_build.gradle'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def genVersionName(def versionName){
 | 
				
			||||||
 | 
					    // 检查编译标志位配置
 | 
				
			||||||
 | 
					    assert (winbollBuildProps['stageCount'] != null)
 | 
				
			||||||
 | 
					    assert (winbollBuildProps['baseVersion'] != null)
 | 
				
			||||||
 | 
					    // 保存基础版本号
 | 
				
			||||||
 | 
					    winbollBuildProps.setProperty("baseVersion", "${versionName}");
 | 
				
			||||||
 | 
					    //保存编译标志配置
 | 
				
			||||||
 | 
					    FileOutputStream fos = new FileOutputStream(winbollBuildPropsFile)
 | 
				
			||||||
 | 
					    winbollBuildProps.store(fos, "${winbollBuildPropsDesc}");
 | 
				
			||||||
 | 
					    fos.close();
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    // 返回编译版本号
 | 
				
			||||||
 | 
					    return "${versionName}." + winbollBuildProps['stageCount']
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					android {
 | 
				
			||||||
 | 
					    compileSdkVersion 32
 | 
				
			||||||
 | 
					    buildToolsVersion "32.0.0"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    defaultConfig {
 | 
				
			||||||
 | 
					        applicationId "cc.winboll.studio.numtable"
 | 
				
			||||||
 | 
					        minSdkVersion 24
 | 
				
			||||||
 | 
					        targetSdkVersion 30
 | 
				
			||||||
 | 
					        versionCode 1
 | 
				
			||||||
 | 
					        // versionName 更新后需要手动设置 
 | 
				
			||||||
 | 
					        // .winboll/winbollBuildProps.properties 文件的 stageCount=0
 | 
				
			||||||
 | 
					        // Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
 | 
				
			||||||
 | 
					        versionName "15.1"
 | 
				
			||||||
 | 
					        if(true) {
 | 
				
			||||||
 | 
					            versionName = genVersionName("${versionName}")
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    buildTypes {
 | 
				
			||||||
 | 
					        release {
 | 
				
			||||||
 | 
					            minifyEnabled false
 | 
				
			||||||
 | 
					            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					dependencies {
 | 
				
			||||||
 | 
					    api fileTree(dir: 'libs', include: ['*.jar'])
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    // SSH
 | 
				
			||||||
 | 
					    api 'com.jcraft:jsch:0.1.55'
 | 
				
			||||||
 | 
					    // Html 解析
 | 
				
			||||||
 | 
					    api 'org.jsoup:jsoup:1.13.1'
 | 
				
			||||||
 | 
					    // 二维码类库
 | 
				
			||||||
 | 
					    api 'com.google.zxing:core:3.4.1'
 | 
				
			||||||
 | 
					    api 'com.journeyapps:zxing-android-embedded:3.6.0'
 | 
				
			||||||
 | 
					    // 应用介绍页类库
 | 
				
			||||||
 | 
					    api 'io.github.medyo:android-about-page:2.0.0'
 | 
				
			||||||
 | 
					    // 吐司类库
 | 
				
			||||||
 | 
					    api 'com.github.getActivity:ToastUtils:10.5'
 | 
				
			||||||
 | 
					    // 网络连接类库
 | 
				
			||||||
 | 
					    api 'com.squareup.okhttp3:okhttp:4.4.1'
 | 
				
			||||||
 | 
					    // AndroidX 类库
 | 
				
			||||||
 | 
					    api 'androidx.appcompat:appcompat:1.1.0'
 | 
				
			||||||
 | 
					    api 'com.google.android.material:material:1.4.0'
 | 
				
			||||||
 | 
					    //api 'androidx.viewpager:viewpager:1.0.0'
 | 
				
			||||||
 | 
					    //api 'androidx.vectordrawable:vectordrawable:1.1.0'
 | 
				
			||||||
 | 
					    //api 'androidx.vectordrawable:vectordrawable-animated:1.1.0'
 | 
				
			||||||
 | 
					    //api 'androidx.fragment:fragment:1.1.0'
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					    api 'cc.winboll.studio:libaes:15.8.0'
 | 
				
			||||||
 | 
					    api 'cc.winboll.studio:libapputils:15.8.2'
 | 
				
			||||||
 | 
					    api 'cc.winboll.studio:libappbase:15.8.2'
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										8
									
								
								numtable/build.properties
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					#Created by .winboll/winboll_app_build.gradle
 | 
				
			||||||
 | 
					#Sun Jun 08 21:21:11 HKT 2025
 | 
				
			||||||
 | 
					stageCount=1
 | 
				
			||||||
 | 
					libraryProject=
 | 
				
			||||||
 | 
					baseVersion=15.1
 | 
				
			||||||
 | 
					publishVersion=15.1.0
 | 
				
			||||||
 | 
					buildCount=0
 | 
				
			||||||
 | 
					baseBetaVersion=15.1.1
 | 
				
			||||||
							
								
								
									
										21
									
								
								numtable/proguard-rules.pro
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					# Add project specific ProGuard rules here.
 | 
				
			||||||
 | 
					# You can control the set of applied configuration files using the
 | 
				
			||||||
 | 
					# proguardFiles setting in build.gradle.
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# For more details, see
 | 
				
			||||||
 | 
					#   http://developer.android.com/guide/developing/tools/proguard.html
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# If your project uses WebView with JS, uncomment the following
 | 
				
			||||||
 | 
					# and specify the fully qualified class name to the JavaScript interface
 | 
				
			||||||
 | 
					# class:
 | 
				
			||||||
 | 
					#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
 | 
				
			||||||
 | 
					#   public *;
 | 
				
			||||||
 | 
					#}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Uncomment this to preserve the line number information for
 | 
				
			||||||
 | 
					# debugging stack traces.
 | 
				
			||||||
 | 
					#-keepattributes SourceFile,LineNumberTable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# If you keep the line number information, uncomment this to
 | 
				
			||||||
 | 
					# hide the original source file name.
 | 
				
			||||||
 | 
					#-renamesourcefileattribute SourceFile
 | 
				
			||||||
							
								
								
									
										14
									
								
								numtable/src/beta/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,14 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<manifest xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    xmlns:tools="http://schemas.android.com/tools" >
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <application
 | 
				
			||||||
 | 
					        tools:replace="android:icon"
 | 
				
			||||||
 | 
					        android:icon="@drawable/ic_launcher_beta">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <!-- Put flavor specific code here -->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    </application>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</manifest>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										6
									
								
								numtable/src/beta/res/values/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,6 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<resources>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <string name="app_name">NumTable +</string>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</resources>
 | 
				
			||||||
							
								
								
									
										37
									
								
								numtable/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,37 @@
 | 
				
			|||||||
 | 
					<?xml version='1.0' encoding='utf-8'?>
 | 
				
			||||||
 | 
					<manifest
 | 
				
			||||||
 | 
					    xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    package="cc.winboll.studio.numtable">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <application
 | 
				
			||||||
 | 
					        android:allowBackup="true"
 | 
				
			||||||
 | 
					        android:icon="@drawable/ic_launcher"
 | 
				
			||||||
 | 
					        android:roundIcon="@drawable/ic_launcher"
 | 
				
			||||||
 | 
					        android:label="@string/app_name"
 | 
				
			||||||
 | 
					        android:theme="@style/MyAppTheme"
 | 
				
			||||||
 | 
					        android:resizeableActivity="true"
 | 
				
			||||||
 | 
					        android:name=".App">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <activity
 | 
				
			||||||
 | 
					            android:name=".MainActivity"
 | 
				
			||||||
 | 
					            android:label="@string/app_name">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            <intent-filter>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                <action android:name="android.intent.action.MAIN"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                <category android:name="android.intent.category.LAUNCHER"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            </intent-filter>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        </activity>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <meta-data
 | 
				
			||||||
 | 
					            android:name="android.max_aspect"
 | 
				
			||||||
 | 
					            android:value="4.0"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <activity android:name=".GlobalApplication$CrashActivity"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    </application>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</manifest>
 | 
				
			||||||
							
								
								
									
										345
									
								
								numtable/src/main/java/cc/winboll/studio/numtable/App.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,345 @@
 | 
				
			|||||||
 | 
					package cc.winboll.studio.numtable;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import android.app.Activity;
 | 
				
			||||||
 | 
					import android.content.ClipData;
 | 
				
			||||||
 | 
					import android.content.ClipboardManager;
 | 
				
			||||||
 | 
					import android.content.Context;
 | 
				
			||||||
 | 
					import android.content.Intent;
 | 
				
			||||||
 | 
					import android.content.pm.PackageInfo;
 | 
				
			||||||
 | 
					import android.content.res.Resources;
 | 
				
			||||||
 | 
					import android.graphics.Typeface;
 | 
				
			||||||
 | 
					import android.os.Build;
 | 
				
			||||||
 | 
					import android.os.Bundle;
 | 
				
			||||||
 | 
					import android.os.Handler;
 | 
				
			||||||
 | 
					import android.os.Looper;
 | 
				
			||||||
 | 
					import android.text.TextUtils;
 | 
				
			||||||
 | 
					import android.util.Log;
 | 
				
			||||||
 | 
					import android.view.Gravity;
 | 
				
			||||||
 | 
					import android.view.Menu;
 | 
				
			||||||
 | 
					import android.view.MenuItem;
 | 
				
			||||||
 | 
					import android.view.ViewGroup;
 | 
				
			||||||
 | 
					import android.widget.HorizontalScrollView;
 | 
				
			||||||
 | 
					import android.widget.ScrollView;
 | 
				
			||||||
 | 
					import android.widget.TextView;
 | 
				
			||||||
 | 
					import android.widget.Toast;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.GlobalApplication;
 | 
				
			||||||
 | 
					import com.hjq.toast.ToastUtils;
 | 
				
			||||||
 | 
					import com.hjq.toast.style.WhiteToastStyle;
 | 
				
			||||||
 | 
					import java.io.ByteArrayInputStream;
 | 
				
			||||||
 | 
					import java.io.ByteArrayOutputStream;
 | 
				
			||||||
 | 
					import java.io.Closeable;
 | 
				
			||||||
 | 
					import java.io.File;
 | 
				
			||||||
 | 
					import java.io.FileInputStream;
 | 
				
			||||||
 | 
					import java.io.FileOutputStream;
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					import java.io.InputStream;
 | 
				
			||||||
 | 
					import java.io.OutputStream;
 | 
				
			||||||
 | 
					import java.lang.Thread.UncaughtExceptionHandler;
 | 
				
			||||||
 | 
					import java.text.DateFormat;
 | 
				
			||||||
 | 
					import java.text.SimpleDateFormat;
 | 
				
			||||||
 | 
					import java.util.Arrays;
 | 
				
			||||||
 | 
					import java.util.Date;
 | 
				
			||||||
 | 
					import java.util.LinkedHashMap;
 | 
				
			||||||
 | 
					import java.util.concurrent.atomic.AtomicBoolean;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class App extends GlobalApplication {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    private static Handler MAIN_HANDLER = new Handler(Looper.getMainLooper());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    public void onCreate() {
 | 
				
			||||||
 | 
					        super.onCreate();
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					        // 初始化 Toast 框架
 | 
				
			||||||
 | 
					        ToastUtils.init(this);
 | 
				
			||||||
 | 
					        // 设置 Toast 布局样式
 | 
				
			||||||
 | 
					        //ToastUtils.setView(R.layout.view_toast);
 | 
				
			||||||
 | 
					        ToastUtils.setStyle(new WhiteToastStyle());
 | 
				
			||||||
 | 
					        ToastUtils.setGravity(Gravity.BOTTOM, 0, 200);
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					        //CrashHandler.getInstance().registerGlobal(this);
 | 
				
			||||||
 | 
					        //CrashHandler.getInstance().registerPart(this);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static void write(InputStream input, OutputStream output) throws IOException {
 | 
				
			||||||
 | 
					        byte[] buf = new byte[1024 * 8];
 | 
				
			||||||
 | 
					        int len;
 | 
				
			||||||
 | 
					        while ((len = input.read(buf)) != -1) {
 | 
				
			||||||
 | 
					            output.write(buf, 0, len);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static void write(File file, byte[] data) throws IOException {
 | 
				
			||||||
 | 
					        File parent = file.getParentFile();
 | 
				
			||||||
 | 
					        if (parent != null && !parent.exists()) parent.mkdirs();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ByteArrayInputStream input = new ByteArrayInputStream(data);
 | 
				
			||||||
 | 
					        FileOutputStream output = new FileOutputStream(file);
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            write(input, output);
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            closeIO(input, output);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static String toString(InputStream input) throws IOException {
 | 
				
			||||||
 | 
					        ByteArrayOutputStream output = new ByteArrayOutputStream();
 | 
				
			||||||
 | 
					        write(input, output);
 | 
				
			||||||
 | 
					        try {
 | 
				
			||||||
 | 
					            return output.toString("UTF-8");
 | 
				
			||||||
 | 
					        } finally {
 | 
				
			||||||
 | 
					            closeIO(input, output);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static void closeIO(Closeable... closeables) {
 | 
				
			||||||
 | 
					        for (Closeable closeable : closeables) {
 | 
				
			||||||
 | 
					            try {
 | 
				
			||||||
 | 
					                if (closeable != null) closeable.close();
 | 
				
			||||||
 | 
					            } catch (IOException ignored) {}
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static class CrashHandler {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public static final UncaughtExceptionHandler DEFAULT_UNCAUGHT_EXCEPTION_HANDLER = Thread.getDefaultUncaughtExceptionHandler();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private static CrashHandler sInstance;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private PartCrashHandler mPartCrashHandler;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public static CrashHandler getInstance() {
 | 
				
			||||||
 | 
					            if (sInstance == null) {
 | 
				
			||||||
 | 
					                sInstance = new CrashHandler();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return sInstance;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public void registerGlobal(Context context) {
 | 
				
			||||||
 | 
					            registerGlobal(context, null);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public void registerGlobal(Context context, String crashDir) {
 | 
				
			||||||
 | 
					            Thread.setDefaultUncaughtExceptionHandler(new UncaughtExceptionHandlerImpl(context.getApplicationContext(), crashDir));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public void unregister() {
 | 
				
			||||||
 | 
					            Thread.setDefaultUncaughtExceptionHandler(DEFAULT_UNCAUGHT_EXCEPTION_HANDLER);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public void registerPart(Context context) {
 | 
				
			||||||
 | 
					            unregisterPart(context);
 | 
				
			||||||
 | 
					            mPartCrashHandler = new PartCrashHandler(context.getApplicationContext());
 | 
				
			||||||
 | 
					            MAIN_HANDLER.postAtFrontOfQueue(mPartCrashHandler);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public void unregisterPart(Context context) {
 | 
				
			||||||
 | 
					            if (mPartCrashHandler != null) {
 | 
				
			||||||
 | 
					                mPartCrashHandler.isRunning.set(false);
 | 
				
			||||||
 | 
					                mPartCrashHandler = null;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private static class PartCrashHandler implements Runnable {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private final Context mContext;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            public AtomicBoolean isRunning = new AtomicBoolean(true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            public PartCrashHandler(Context context) {
 | 
				
			||||||
 | 
					                this.mContext = context;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            @Override
 | 
				
			||||||
 | 
					            public void run() {
 | 
				
			||||||
 | 
					                while (isRunning.get()) {
 | 
				
			||||||
 | 
					                    try {
 | 
				
			||||||
 | 
					                        Looper.loop();
 | 
				
			||||||
 | 
					                    } catch (final Throwable e) {
 | 
				
			||||||
 | 
					                        e.printStackTrace();
 | 
				
			||||||
 | 
					                        if (isRunning.get()) {
 | 
				
			||||||
 | 
					                            MAIN_HANDLER.post(new Runnable(){
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                                    @Override
 | 
				
			||||||
 | 
					                                    public void run() {
 | 
				
			||||||
 | 
					                                        Toast.makeText(mContext, e.toString(), Toast.LENGTH_LONG).show();
 | 
				
			||||||
 | 
					                                    }
 | 
				
			||||||
 | 
					                                });
 | 
				
			||||||
 | 
					                        } else {
 | 
				
			||||||
 | 
					                            if (e instanceof RuntimeException) {
 | 
				
			||||||
 | 
					                                throw (RuntimeException)e;
 | 
				
			||||||
 | 
					                            } else {
 | 
				
			||||||
 | 
					                                throw new RuntimeException(e);
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private static class UncaughtExceptionHandlerImpl implements UncaughtExceptionHandler {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private static DateFormat DATE_FORMAT = new SimpleDateFormat("yyyy_MM_dd-HH_mm_ss");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private final Context mContext;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private final File mCrashDir;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            public UncaughtExceptionHandlerImpl(Context context, String crashDir) {
 | 
				
			||||||
 | 
					                this.mContext = context;
 | 
				
			||||||
 | 
					                this.mCrashDir = TextUtils.isEmpty(crashDir) ? new File(mContext.getExternalCacheDir(), "crash") : new File(crashDir);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            @Override
 | 
				
			||||||
 | 
					            public void uncaughtException(Thread thread, Throwable throwable) {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    String log = buildLog(throwable);
 | 
				
			||||||
 | 
					                    writeLog(log);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    try {
 | 
				
			||||||
 | 
					                        Intent intent = new Intent(mContext, CrashActivity.class);
 | 
				
			||||||
 | 
					                        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
 | 
				
			||||||
 | 
					                        intent.putExtra(Intent.EXTRA_TEXT, log);
 | 
				
			||||||
 | 
					                        mContext.startActivity(intent);
 | 
				
			||||||
 | 
					                    } catch (Throwable e) {
 | 
				
			||||||
 | 
					                        e.printStackTrace();
 | 
				
			||||||
 | 
					                        writeLog(e.toString());
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                    throwable.printStackTrace();
 | 
				
			||||||
 | 
					                    android.os.Process.killProcess(android.os.Process.myPid());
 | 
				
			||||||
 | 
					                    System.exit(0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                } catch (Throwable e) {
 | 
				
			||||||
 | 
					                    if (DEFAULT_UNCAUGHT_EXCEPTION_HANDLER != null) DEFAULT_UNCAUGHT_EXCEPTION_HANDLER.uncaughtException(thread, throwable);
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private String buildLog(Throwable throwable) {
 | 
				
			||||||
 | 
					                String time = DATE_FORMAT.format(new Date());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                String versionName = "unknown";
 | 
				
			||||||
 | 
					                long versionCode = 0;
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    PackageInfo packageInfo = mContext.getPackageManager().getPackageInfo(mContext.getPackageName(), 0);
 | 
				
			||||||
 | 
					                    versionName = packageInfo.versionName;
 | 
				
			||||||
 | 
					                    versionCode = Build.VERSION.SDK_INT >= 28 ? packageInfo.getLongVersionCode() : packageInfo.versionCode;
 | 
				
			||||||
 | 
					                } catch (Throwable ignored) {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                LinkedHashMap<String, String> head = new LinkedHashMap<String, String>();
 | 
				
			||||||
 | 
					                head.put("Time Of Crash", time);
 | 
				
			||||||
 | 
					                head.put("Device", String.format("%s, %s", Build.MANUFACTURER, Build.MODEL));
 | 
				
			||||||
 | 
					                head.put("Android Version", String.format("%s (%d)", Build.VERSION.RELEASE, Build.VERSION.SDK_INT));
 | 
				
			||||||
 | 
					                head.put("App Version", String.format("%s (%d)", versionName, versionCode));
 | 
				
			||||||
 | 
					                head.put("Kernel", getKernel());
 | 
				
			||||||
 | 
					                head.put("Support Abis", Build.VERSION.SDK_INT >= 21 && Build.SUPPORTED_ABIS != null ? Arrays.toString(Build.SUPPORTED_ABIS): "unknown");
 | 
				
			||||||
 | 
					                head.put("Fingerprint", Build.FINGERPRINT);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                StringBuilder builder = new StringBuilder();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                for (String key : head.keySet()) {
 | 
				
			||||||
 | 
					                    if (builder.length() != 0) builder.append("\n");
 | 
				
			||||||
 | 
					                    builder.append(key);
 | 
				
			||||||
 | 
					                    builder.append(" :    ");
 | 
				
			||||||
 | 
					                    builder.append(head.get(key));
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                builder.append("\n\n");
 | 
				
			||||||
 | 
					                builder.append(Log.getStackTraceString(throwable));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                return builder.toString(); 
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private void writeLog(String log) {
 | 
				
			||||||
 | 
					                String time = DATE_FORMAT.format(new Date());
 | 
				
			||||||
 | 
					                File file = new File(mCrashDir, "crash_" + time + ".txt");
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    write(file, log.getBytes("UTF-8"));
 | 
				
			||||||
 | 
					                } catch (Throwable e) {
 | 
				
			||||||
 | 
					                    e.printStackTrace();
 | 
				
			||||||
 | 
					                } 
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            private static String getKernel() {
 | 
				
			||||||
 | 
					                try {
 | 
				
			||||||
 | 
					                    return App.toString(new FileInputStream("/proc/version")).trim();
 | 
				
			||||||
 | 
					                } catch (Throwable e) {
 | 
				
			||||||
 | 
					                    return e.getMessage();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public static final class CrashActivity extends Activity {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private String mLog;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        @Override
 | 
				
			||||||
 | 
					        protected void onCreate(Bundle savedInstanceState) {
 | 
				
			||||||
 | 
					            super.onCreate(savedInstanceState);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            setTheme(android.R.style.Theme_DeviceDefault);
 | 
				
			||||||
 | 
					            setTitle("App Crash");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            mLog = getIntent().getStringExtra(Intent.EXTRA_TEXT);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            ScrollView contentView = new ScrollView(this);
 | 
				
			||||||
 | 
					            contentView.setFillViewport(true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            HorizontalScrollView horizontalScrollView = new HorizontalScrollView(this);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            TextView textView = new TextView(this);
 | 
				
			||||||
 | 
					            int padding = dp2px(16);
 | 
				
			||||||
 | 
					            textView.setPadding(padding, padding, padding, padding);
 | 
				
			||||||
 | 
					            textView.setText(mLog);
 | 
				
			||||||
 | 
					            textView.setTextIsSelectable(true);
 | 
				
			||||||
 | 
					            textView.setTypeface(Typeface.DEFAULT);
 | 
				
			||||||
 | 
					            textView.setLinksClickable(true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            horizontalScrollView.addView(textView);
 | 
				
			||||||
 | 
					            contentView.addView(horizontalScrollView, ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            setContentView(contentView);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private void restart() {
 | 
				
			||||||
 | 
					            Intent intent = getPackageManager().getLaunchIntentForPackage(getPackageName());
 | 
				
			||||||
 | 
					            if (intent != null) {
 | 
				
			||||||
 | 
					                intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
 | 
				
			||||||
 | 
					                startActivity(intent);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            finish();
 | 
				
			||||||
 | 
					            android.os.Process.killProcess(android.os.Process.myPid());
 | 
				
			||||||
 | 
					            System.exit(0);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private static int dp2px(float dpValue) {
 | 
				
			||||||
 | 
					            final float scale = Resources.getSystem().getDisplayMetrics().density;
 | 
				
			||||||
 | 
					            return (int) (dpValue * scale + 0.5f);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        @Override
 | 
				
			||||||
 | 
					        public boolean onCreateOptionsMenu(Menu menu) {
 | 
				
			||||||
 | 
					            menu.add(0, android.R.id.copy, 0, android.R.string.copy)
 | 
				
			||||||
 | 
					                .setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM);
 | 
				
			||||||
 | 
					            return super.onCreateOptionsMenu(menu);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        @Override
 | 
				
			||||||
 | 
					        public boolean onOptionsItemSelected(MenuItem item) {
 | 
				
			||||||
 | 
					            switch (item.getItemId()) {
 | 
				
			||||||
 | 
					                case android.R.id.copy:
 | 
				
			||||||
 | 
					                    ClipboardManager cm = (ClipboardManager) getSystemService(Context.CLIPBOARD_SERVICE);
 | 
				
			||||||
 | 
					                    cm.setPrimaryClip(ClipData.newPlainText(getPackageName(), mLog));
 | 
				
			||||||
 | 
					                    return true;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return super.onOptionsItemSelected(item);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        @Override
 | 
				
			||||||
 | 
					        public void onBackPressed() {
 | 
				
			||||||
 | 
					            restart();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -0,0 +1,31 @@
 | 
				
			|||||||
 | 
					package cc.winboll.studio.numtable;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import android.os.Bundle;
 | 
				
			||||||
 | 
					import androidx.appcompat.app.AppCompatActivity;
 | 
				
			||||||
 | 
					import androidx.appcompat.widget.Toolbar;
 | 
				
			||||||
 | 
					import cc.winboll.studio.libappbase.LogView;
 | 
				
			||||||
 | 
					import com.hjq.toast.ToastUtils;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public class MainActivity extends AppCompatActivity {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    LogView mLogView;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    protected void onCreate(Bundle savedInstanceState) {
 | 
				
			||||||
 | 
					        super.onCreate(savedInstanceState);
 | 
				
			||||||
 | 
					        setContentView(R.layout.activity_main);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							Toolbar toolbar=(Toolbar)findViewById(R.id.toolbar);
 | 
				
			||||||
 | 
							setSupportActionBar(toolbar);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mLogView = findViewById(R.id.logview);
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					        ToastUtils.show("onCreate");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @Override
 | 
				
			||||||
 | 
					    protected void onResume() {
 | 
				
			||||||
 | 
					        super.onResume();
 | 
				
			||||||
 | 
					        mLogView.start();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -0,0 +1,34 @@
 | 
				
			|||||||
 | 
					<vector xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    xmlns:aapt="http://schemas.android.com/aapt"
 | 
				
			||||||
 | 
					    android:width="108dp"
 | 
				
			||||||
 | 
					    android:height="108dp"
 | 
				
			||||||
 | 
					    android:viewportHeight="108"
 | 
				
			||||||
 | 
					    android:viewportWidth="108">
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillType="evenOdd"
 | 
				
			||||||
 | 
					        android:pathData="M32,64C32,64 38.39,52.99 44.13,50.95C51.37,48.37 70.14,49.57 70.14,49.57L108.26,87.69L108,109.01L75.97,107.97L32,64Z"
 | 
				
			||||||
 | 
					        android:strokeColor="#00000000"
 | 
				
			||||||
 | 
					        android:strokeWidth="1">
 | 
				
			||||||
 | 
					        <aapt:attr name="android:fillColor">
 | 
				
			||||||
 | 
					            <gradient
 | 
				
			||||||
 | 
					                android:endX="78.5885"
 | 
				
			||||||
 | 
					                android:endY="90.9159"
 | 
				
			||||||
 | 
					                android:startX="48.7653"
 | 
				
			||||||
 | 
					                android:startY="61.0927"
 | 
				
			||||||
 | 
					                android:type="linear">
 | 
				
			||||||
 | 
					                <item
 | 
				
			||||||
 | 
					                    android:color="#44000000"
 | 
				
			||||||
 | 
					                    android:offset="0.0" />
 | 
				
			||||||
 | 
					                <item
 | 
				
			||||||
 | 
					                    android:color="#00000000"
 | 
				
			||||||
 | 
					                    android:offset="1.0" />
 | 
				
			||||||
 | 
					            </gradient>
 | 
				
			||||||
 | 
					        </aapt:attr>
 | 
				
			||||||
 | 
					    </path>
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#FFFFFF"
 | 
				
			||||||
 | 
					        android:fillType="nonZero"
 | 
				
			||||||
 | 
					        android:pathData="M66.94,46.02L66.94,46.02C72.44,50.07 76,56.61 76,64L32,64C32,56.61 35.56,50.11 40.98,46.06L36.18,41.19C35.45,40.45 35.45,39.3 36.18,38.56C36.91,37.81 38.05,37.81 38.78,38.56L44.25,44.05C47.18,42.57 50.48,41.71 54,41.71C57.48,41.71 60.78,42.57 63.68,44.05L69.11,38.56C69.84,37.81 70.98,37.81 71.71,38.56C72.44,39.3 72.44,40.45 71.71,41.19L66.94,46.02ZM62.94,56.92C64.08,56.92 65,56.01 65,54.88C65,53.76 64.08,52.85 62.94,52.85C61.8,52.85 60.88,53.76 60.88,54.88C60.88,56.01 61.8,56.92 62.94,56.92ZM45.06,56.92C46.2,56.92 47.13,56.01 47.13,54.88C47.13,53.76 46.2,52.85 45.06,52.85C43.92,52.85 43,53.76 43,54.88C43,56.01 43.92,56.92 45.06,56.92Z"
 | 
				
			||||||
 | 
					        android:strokeColor="#00000000"
 | 
				
			||||||
 | 
					        android:strokeWidth="1" />
 | 
				
			||||||
 | 
					</vector>
 | 
				
			||||||
							
								
								
									
										13
									
								
								numtable/src/main/res/drawable/ic_launcher.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,13 @@
 | 
				
			|||||||
 | 
					<vector xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    android:width="512dp"
 | 
				
			||||||
 | 
					    android:height="512dp"
 | 
				
			||||||
 | 
					    android:viewportWidth="512"
 | 
				
			||||||
 | 
					    android:viewportHeight="512">
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#FF3D8A1C"
 | 
				
			||||||
 | 
					        android:strokeColor="#FFF5DD00"
 | 
				
			||||||
 | 
					        android:strokeWidth="20.0"
 | 
				
			||||||
 | 
					        android:strokeLineCap="round"
 | 
				
			||||||
 | 
					        android:strokeMiterLimit="10"
 | 
				
			||||||
 | 
					        android:pathData="M47.53 417.75C38.23 258.17 97.09 94.95 156.27 94.16 211.66 98.38 260.9 237.99 329.08 242.47 351.54 241.69 397.94 228.17 449.46 88.59 447.52 262.11 394.97 417.45 338.71 418.74 279.06 412.93 220.17 297.22 157.81 295.26 132.98 294.7 116.37 308.69 47.53 417.75Z"/>
 | 
				
			||||||
 | 
					</vector>
 | 
				
			||||||
							
								
								
									
										170
									
								
								numtable/src/main/res/drawable/ic_launcher_background.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,170 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<vector xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    android:width="108dp"
 | 
				
			||||||
 | 
					    android:height="108dp"
 | 
				
			||||||
 | 
					    android:viewportHeight="108"
 | 
				
			||||||
 | 
					    android:viewportWidth="108">
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#FFFBC41E"
 | 
				
			||||||
 | 
					        android:pathData="M0,0h108v108h-108z" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M9,0L9,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,0L19,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M29,0L29,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M39,0L39,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M49,0L49,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M59,0L59,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M69,0L69,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M79,0L79,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M89,0L89,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M99,0L99,108"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,9L108,9"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,19L108,19"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,29L108,29"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,39L108,39"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,49L108,49"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,59L108,59"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,69L108,69"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,79L108,79"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,89L108,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M0,99L108,99"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,29L89,29"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,39L89,39"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,49L89,49"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,59L89,59"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,69L89,69"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M19,79L89,79"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M29,19L29,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M39,19L39,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M49,19L49,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M59,19L59,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M69,19L69,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					    <path
 | 
				
			||||||
 | 
					        android:fillColor="#00000000"
 | 
				
			||||||
 | 
					        android:pathData="M79,19L79,89"
 | 
				
			||||||
 | 
					        android:strokeColor="#33FFFFFF"
 | 
				
			||||||
 | 
					        android:strokeWidth="0.8" />
 | 
				
			||||||
 | 
					</vector>
 | 
				
			||||||
							
								
								
									
										11
									
								
								numtable/src/main/res/drawable/ic_launcher_beta.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,11 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<layer-list xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    android:clickable="true">
 | 
				
			||||||
 | 
					    <item android:drawable="@drawable/ic_launcher_background"/> 
 | 
				
			||||||
 | 
					    <item 
 | 
				
			||||||
 | 
					        android:left="15dp" 
 | 
				
			||||||
 | 
					        android:top="15dp" 
 | 
				
			||||||
 | 
					        android:right="15dp" 
 | 
				
			||||||
 | 
					        android:bottom="15dp"
 | 
				
			||||||
 | 
					        android:drawable="@drawable/ic_launcher"/>
 | 
				
			||||||
 | 
					</layer-list>
 | 
				
			||||||
							
								
								
									
										51
									
								
								numtable/src/main/res/layout/activity_main.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,51 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<LinearLayout
 | 
				
			||||||
 | 
						xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
						xmlns:app="http://schemas.android.com/apk/res-auto"
 | 
				
			||||||
 | 
						android:layout_width="match_parent"
 | 
				
			||||||
 | 
						android:layout_height="match_parent"
 | 
				
			||||||
 | 
						android:orientation="vertical">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<com.google.android.material.appbar.AppBarLayout
 | 
				
			||||||
 | 
							android:layout_width="match_parent"
 | 
				
			||||||
 | 
							android:layout_height="wrap_content"
 | 
				
			||||||
 | 
							android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							<androidx.appcompat.widget.Toolbar
 | 
				
			||||||
 | 
								android:id="@+id/toolbar"
 | 
				
			||||||
 | 
								android:layout_width="match_parent"
 | 
				
			||||||
 | 
								android:layout_height="?attr/actionBarSize"
 | 
				
			||||||
 | 
								app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						</com.google.android.material.appbar.AppBarLayout>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<LinearLayout
 | 
				
			||||||
 | 
							android:orientation="vertical"
 | 
				
			||||||
 | 
							android:layout_width="match_parent"
 | 
				
			||||||
 | 
							android:layout_height="0dp"
 | 
				
			||||||
 | 
							android:layout_weight="1.0"
 | 
				
			||||||
 | 
							android:gravity="center_vertical|center_horizontal">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							<TextView
 | 
				
			||||||
 | 
								android:layout_width="wrap_content"
 | 
				
			||||||
 | 
								android:layout_height="wrap_content"
 | 
				
			||||||
 | 
								android:text="NumTable"
 | 
				
			||||||
 | 
								android:textAppearance="?android:attr/textAppearanceLarge"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						</LinearLayout>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<LinearLayout
 | 
				
			||||||
 | 
							android:orientation="vertical"
 | 
				
			||||||
 | 
							android:layout_width="match_parent"
 | 
				
			||||||
 | 
							android:layout_height="0dp"
 | 
				
			||||||
 | 
							android:layout_weight="1.0">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							<cc.winboll.studio.libappbase.LogView
 | 
				
			||||||
 | 
								android:layout_width="match_parent"
 | 
				
			||||||
 | 
								android:layout_height="match_parent"
 | 
				
			||||||
 | 
								android:id="@+id/logview"/>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						</LinearLayout>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</LinearLayout>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										5
									
								
								numtable/src/main/res/mipmap-anydpi-v26/ic_launcher.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,5 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
 | 
				
			||||||
 | 
					    <background android:drawable="@drawable/ic_launcher_background" />
 | 
				
			||||||
 | 
					    <foreground android:drawable="@drawable/ic_launcher_foreground" />
 | 
				
			||||||
 | 
					</adaptive-icon>
 | 
				
			||||||
@@ -0,0 +1,5 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
 | 
				
			||||||
 | 
					    <background android:drawable="@drawable/ic_launcher_background" />
 | 
				
			||||||
 | 
					    <foreground android:drawable="@drawable/ic_launcher_foreground" />
 | 
				
			||||||
 | 
					</adaptive-icon>
 | 
				
			||||||
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-hdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 3.0 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-hdpi/ic_launcher_round.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 4.9 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-mdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 2.0 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-mdpi/ic_launcher_round.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 2.8 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-xhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 4.5 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-xhdpi/ic_launcher_round.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 6.9 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-xxhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 6.3 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-xxhdpi/ic_launcher_round.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 10 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-xxxhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 9.0 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								numtable/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 15 KiB  | 
							
								
								
									
										6
									
								
								numtable/src/main/res/values/colors.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,6 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<resources>
 | 
				
			||||||
 | 
					    <color name="colorPrimary">#009688</color>
 | 
				
			||||||
 | 
					    <color name="colorPrimaryDark">#00796B</color>
 | 
				
			||||||
 | 
					    <color name="colorAccent">#FF9800</color>
 | 
				
			||||||
 | 
					</resources>
 | 
				
			||||||
							
								
								
									
										4
									
								
								numtable/src/main/res/values/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,4 @@
 | 
				
			|||||||
 | 
					<resources>
 | 
				
			||||||
 | 
					    <string name="app_name">NumTable</string>
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					</resources>
 | 
				
			||||||
							
								
								
									
										11
									
								
								numtable/src/main/res/values/styles.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,11 @@
 | 
				
			|||||||
 | 
					<resources>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <!-- Base application theme. -->
 | 
				
			||||||
 | 
					    <style name="MyAppTheme" parent="Theme.AppCompat.Light.NoActionBar">
 | 
				
			||||||
 | 
					        <!-- Customize your theme here. -->
 | 
				
			||||||
 | 
					        <item name="colorPrimary">@color/colorPrimary</item>
 | 
				
			||||||
 | 
					        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
 | 
				
			||||||
 | 
					        <item name="colorAccent">@color/colorAccent</item>
 | 
				
			||||||
 | 
					    </style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</resources>
 | 
				
			||||||
							
								
								
									
										12
									
								
								numtable/src/stage/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,12 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<manifest xmlns:android="http://schemas.android.com/apk/res/android"
 | 
				
			||||||
 | 
					    xmlns:tools="http://schemas.android.com/tools" >
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <application>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        <!-- Put flavor specific code here -->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    </application>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</manifest>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										6
									
								
								numtable/src/stage/res/values/strings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,6 @@
 | 
				
			|||||||
 | 
					<?xml version="1.0" encoding="utf-8"?>
 | 
				
			||||||
 | 
					<resources>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <!-- Put flavor specific strings here -->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</resources>
 | 
				
			||||||
@@ -52,3 +52,7 @@
 | 
				
			|||||||
// Ollama 项目编译设置
 | 
					// Ollama 项目编译设置
 | 
				
			||||||
//include ':ollama'
 | 
					//include ':ollama'
 | 
				
			||||||
//rootProject.name = "ollama"
 | 
					//rootProject.name = "ollama"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NumTable 项目编译设置
 | 
				
			||||||
 | 
					//include ':numtable'
 | 
				
			||||||
 | 
					//rootProject.name = "numtable"
 | 
				
			||||||
 
 | 
				
			|||||||