diff --git a/positions/build.gradle b/positions/build.gradle index e4f159b..5b4ea14 100644 --- a/positions/build.gradle +++ b/positions/build.gradle @@ -18,29 +18,34 @@ def genVersionName(def versionName){ } android { - // 1. compileSdkVersion:必须 ≥ targetSdkVersion,建议直接等于 targetSdkVersion(30) - compileSdkVersion 30 + compileSdkVersion 32 + buildToolsVersion "32.0.0" - // 2. buildToolsVersion:需匹配 compileSdkVersion,建议使用 30.x.x 最新稳定版(无需高于 compileSdkVersion) - buildToolsVersion "30.0.3" // 这是 30 对应的最新稳定版,避免使用 beta 版 - defaultConfig { applicationId "cc.winboll.studio.positions" - minSdkVersion 23 + minSdkVersion 24 targetSdkVersion 30 versionCode 1 // versionName 更新后需要手动设置 // .winboll/winbollBuildProps.properties 文件的 stageCount=0 // Gradle编译环境下合起来的 versionName 就是 "${versionName}.0" - versionName "15.12" + versionName "15.0" if(true) { versionName = genVersionName("${versionName}") } } - + + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } } dependencies { + api fileTree(dir: 'libs', include: ['*.jar']) + // https://mvnrepository.com/artifact/com.jzxiang.pickerview/TimePickerDialog api 'com.jzxiang.pickerview:TimePickerDialog:1.0.1' @@ -56,6 +61,8 @@ dependencies { 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 类库 @@ -66,14 +73,7 @@ dependencies { //api 'androidx.vectordrawable:vectordrawable-animated:1.1.0' //api 'androidx.fragment:fragment:1.1.0' - - // WinBoLL库 nexus.winboll.cc 地址 - //api 'cc.winboll.studio:libaes:15.12.0' - //api 'cc.winboll.studio:libappbase:15.12.2' - - // WinBoLL备用库 jitpack.io 地址 - api 'com.github.ZhanGSKen:AES:aes-v15.12.3' - api 'com.github.ZhanGSKen:APPBase:appbase-v15.12.2' - - api fileTree(dir: 'libs', include: ['*.jar']) + api 'cc.winboll.studio:libaes:15.10.2' + api 'cc.winboll.studio:libapputils:15.10.2' + api 'cc.winboll.studio:libappbase:15.10.9' } diff --git a/positions/build.properties b/positions/build.properties index 33ec84f..0c9f1cc 100644 --- a/positions/build.properties +++ b/positions/build.properties @@ -1,8 +1,8 @@ #Created by .winboll/winboll_app_build.gradle -#Sun Dec 07 11:41:29 GMT 2025 -stageCount=0 +#Tue Oct 28 14:17:09 HKT 2025 +stageCount=17 libraryProject= -baseVersion=15.12 -publishVersion=15.12.0 -buildCount=3 -baseBetaVersion=15.12.1 +baseVersion=15.0 +publishVersion=15.0.16 +buildCount=0 +baseBetaVersion=15.0.17 diff --git a/positions/src/beta/res/values-zh/strings.xml b/positions/src/beta/res/values-zh/strings.xml index 394a306..1e0577a 100644 --- a/positions/src/beta/res/values-zh/strings.xml +++ b/positions/src/beta/res/values-zh/strings.xml @@ -1,5 +1,4 @@ 悟空笔记# - 时空任务# diff --git a/positions/src/beta/res/values/strings.xml b/positions/src/beta/res/values/strings.xml index 5984bfd..5dc93b9 100644 --- a/positions/src/beta/res/values/strings.xml +++ b/positions/src/beta/res/values/strings.xml @@ -1,7 +1,6 @@ - Positions - PositionsPlus+ + Positions + diff --git a/positions/src/beta/res/xml/shortcutsmain.xml b/positions/src/beta/res/xml/shortcutsmain.xml deleted file mode 100644 index 08d8f99..0000000 --- a/positions/src/beta/res/xml/shortcutsmain.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - - - - - - diff --git a/positions/src/beta/res/xml/shortcutsplus.xml b/positions/src/beta/res/xml/shortcutsplus.xml deleted file mode 100644 index 78616e4..0000000 --- a/positions/src/beta/res/xml/shortcutsplus.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - - - - - - diff --git a/positions/src/main/AndroidManifest.xml b/positions/src/main/AndroidManifest.xml index d22e9d1..d302210 100644 --- a/positions/src/main/AndroidManifest.xml +++ b/positions/src/main/AndroidManifest.xml @@ -3,32 +3,20 @@ xmlns:android="http://schemas.android.com/apk/res/android" package="cc.winboll.studio.positions"> + + + - - - - - - - - - - - - - - - - - + + + android:label="@string/app_name"> - + - - - - - - - - - + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + - + - - - - - - - - - - - - - - - - - - - - - + + diff --git a/positions/src/main/java/cc/winboll/studio/positions/App.java b/positions/src/main/java/cc/winboll/studio/positions/App.java index 474212d..c99a5d8 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/App.java +++ b/positions/src/main/java/cc/winboll/studio/positions/App.java @@ -14,6 +14,7 @@ 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; @@ -24,7 +25,6 @@ import android.widget.Toast; import cc.winboll.studio.libaes.utils.WinBoLLActivityManager; import cc.winboll.studio.libappbase.GlobalApplication; import cc.winboll.studio.libappbase.ToastUtils; -import cc.winboll.studio.positions.utils.MyActivityLifecycleCallbacks; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.Closeable; @@ -44,36 +44,24 @@ import java.util.concurrent.atomic.AtomicBoolean; public class App extends GlobalApplication { - public static volatile AppLevel _mAppLevel = AppLevel.WUKONG; - - public static final String COMPONENT_WUKONG = "cc.winboll.studio.positions.MainActivityWukong"; - public static final String COMPONENT_LAOJUN = "cc.winboll.studio.positions.MainActivityLaojun"; - public static final String ACTION_OPEN_APPPLUS = "cc.winboll.studio.positions.App.ACTION_OPEN_APPPLUS"; - public static final String ACTION_CLOSE_APPPLUS = "cc.winboll.studio.positions.App.ACTION_CLOSE_APPPLUS"; - private static Handler MAIN_HANDLER = new Handler(Looper.getMainLooper()); - MyActivityLifecycleCallbacks mMyActivityLifecycleCallbacks; - @Override public void onCreate() { super.onCreate(); - - setIsDebugging(BuildConfig.DEBUG); - + setIsDebuging(BuildConfig.DEBUG); + WinBoLLActivityManager.init(this); - + // 初始化 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); - mMyActivityLifecycleCallbacks = new MyActivityLifecycleCallbacks(); - registerActivityLifecycleCallbacks(mMyActivityLifecycleCallbacks); } public static void write(InputStream input, OutputStream output) throws IOException { diff --git a/positions/src/main/java/cc/winboll/studio/positions/AppLevel.java b/positions/src/main/java/cc/winboll/studio/positions/AppLevel.java deleted file mode 100644 index 40bc3e5..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/AppLevel.java +++ /dev/null @@ -1,43 +0,0 @@ -package cc.winboll.studio.positions; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/10 07:23 - * @Describe 应用级别类型枚举 - */ -public enum AppLevel { - WUKONG("wukong", "悟空级别"), - LAOJUN("laojun", "老君级别"); - - public static final String TAG = "AppLevel"; - - // 枚举属性 - private final String code; // 编码(如 "wukong") - private final String desc; // 描述 - - // 构造方法(Java 7 需显式定义) - AppLevel(String code, String desc) { - this.code = code; - this.desc = desc; - } - - // Getter 方法(获取枚举属性) - public String getCode() { - return code; - } - - public String getDesc() { - return desc; - } - - // 可选:根据 code 获取枚举项(便于业务使用) - public static AppLevel getByCode(String code) { - for (AppLevel level : values()) { - if (level.code.equals(code)) { - return level; - } - } - return null; // 或抛出异常,根据业务需求调整 - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/MainActivity.java b/positions/src/main/java/cc/winboll/studio/positions/MainActivity.java index e659b96..d105d63 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/MainActivity.java +++ b/positions/src/main/java/cc/winboll/studio/positions/MainActivity.java @@ -6,7 +6,6 @@ import android.content.Intent; import android.content.pm.PackageManager; import android.os.Build; import android.os.Bundle; -import android.util.Log; import android.view.View; import android.widget.Button; import android.widget.CompoundButton; @@ -18,12 +17,10 @@ import androidx.core.content.ContextCompat; import cc.winboll.studio.libaes.interfaces.IWinBoLLActivity; import cc.winboll.studio.libaes.utils.WinBoLLActivityManager; import cc.winboll.studio.libappbase.LogUtils; -import cc.winboll.studio.libappbase.ToastUtils; import cc.winboll.studio.positions.activities.LocationActivity; import cc.winboll.studio.positions.activities.WinBoLLActivity; -import cc.winboll.studio.positions.utils.APPPlusUtils; +import cc.winboll.studio.positions.services.MainService; import cc.winboll.studio.positions.utils.AppConfigsUtil; -import cc.winboll.studio.positions.utils.JsonShareHandler; import cc.winboll.studio.positions.utils.ServiceUtil; /** @@ -34,18 +31,17 @@ import cc.winboll.studio.positions.utils.ServiceUtil; */ public class MainActivity extends WinBoLLActivity implements IWinBoLLActivity { public static final String TAG = "MainActivity"; - - // 权限请求码(建议定义为类常量,避免魔法值) + // 权限请求码(建议定义为类常量,避免魔法值) private static final int REQUEST_LOCATION_PERMISSIONS = 1001; private static final int REQUEST_BACKGROUND_LOCATION_PERMISSION = 1002; - + // UI 控件:服务控制开关、顶部工具栏 private Switch mServiceSwitch; private Button mManagePositionsButton; private Toolbar mToolbar; // 服务相关:服务实例、绑定状态标记 //private DistanceRefreshService mDistanceService; - //private boolean isServiceBound = false; + private boolean isServiceBound = false; @Override @@ -87,10 +83,6 @@ public class MainActivity extends WinBoLLActivity implements IWinBoLLActivity { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 关联主页面布局 - // 处理启动时的分享 Intent - handleShareIntent(getIntent()); - - // 1. 初始化顶部 Toolbar(保留原逻辑,设置页面标题) initToolbar(); // 2. 初始化其他控件 @@ -103,31 +95,6 @@ public class MainActivity extends WinBoLLActivity implements IWinBoLLActivity { //bindDistanceService(); } - - @Override - protected void onNewIntent(Intent intent) { - super.onNewIntent(intent); - // 处理后续接收的分享 Intent(如应用已在后台) - handleShareIntent(intent); - } - - private void handleShareIntent(Intent intent) { - if (intent != null && Intent.ACTION_SEND.equals(intent.getAction())) { - // 调用工具类,弹出确认对话框 - JsonShareHandler.handleSharedJsonWithConfirm(this, intent, new JsonShareHandler.ConfirmCallback() { - @Override - public void onConfirm(boolean isConfirm) { - // 回调处理:isConfirm 为 true 表示接收并保存,false 表示取消 - if (!isConfirm) { - Log.d("MainActivity", "用户取消接收文件"); - // 可添加取消后的逻辑(如关闭页面) - // finish(); - } - } - }); - } - } - @Override protected void onDestroy() { super.onDestroy(); @@ -147,9 +114,9 @@ public class MainActivity extends WinBoLLActivity implements IWinBoLLActivity { mToolbar = (Toolbar) findViewById(R.id.toolbar); // Java 7 显式 findViewById + 强转 setSupportActionBar(mToolbar); // 给ActionBar设置标题(先判断非空,避免空指针异常) - AppLevel appLevel = AppConfigsUtil.getInstance(getApplicationContext()).getAppLevel(true); - getSupportActionBar().setTitle(getString(R.string.app_name)); - + if (getSupportActionBar() != null) { + getSupportActionBar().setTitle(getString(R.string.app_name)); + } } /** diff --git a/positions/src/main/java/cc/winboll/studio/positions/MainActivityLaojun.java b/positions/src/main/java/cc/winboll/studio/positions/MainActivityLaojun.java deleted file mode 100644 index 15b5da5..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/MainActivityLaojun.java +++ /dev/null @@ -1,22 +0,0 @@ -package cc.winboll.studio.positions; -import android.os.Bundle; -import cc.winboll.studio.libappbase.LogUtils; -import cc.winboll.studio.libappbase.ToastUtils; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/13 15:21 - * @Describe MainActivityLaojun - */ -public class MainActivityLaojun extends MainActivity { - - public static final String TAG = "MainActivityLaojun"; - - @Override - protected void onCreate(Bundle savedInstanceState) { - ToastUtils.show("道法自然"); - LogUtils.d(TAG, "玩法归臻"); - super.onCreate(savedInstanceState); - } - -} diff --git a/positions/src/main/java/cc/winboll/studio/positions/PointLevel.java b/positions/src/main/java/cc/winboll/studio/positions/PointLevel.java deleted file mode 100644 index 3c78f4d..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/PointLevel.java +++ /dev/null @@ -1,43 +0,0 @@ -package cc.winboll.studio.positions; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/15 15:14 - * @Describe 应用入口级别类型枚举 - */ -public enum PointLevel { - DORAEMON("doraemon", "叮铛级别"), - WUKONG("wukong", "悟空级别"), - LAOJUN("laojun", "老君级别"); - - public static final String TAG = "PointLevel"; - - // 枚举属性 - private final String code; // 编码(如 "wukong") - private final String desc; // 描述 - - // 构造方法(Java 7 需显式定义) - PointLevel(String code, String desc) { - this.code = code; - this.desc = desc; - } - - // Getter 方法(获取枚举属性) - public String getCode() { - return code; - } - - public String getDesc() { - return desc; - } - - // 可选:根据 code 获取枚举项(便于业务使用) - public static PointLevel getByCode(String code) { - for (PointLevel level : values()) { - if (level.code.equals(code)) { - return level; - } - } - return null; // 或抛出异常,根据业务需求调整 - } -} diff --git a/positions/src/main/java/cc/winboll/studio/positions/activities/ShortcutActionActivity.java b/positions/src/main/java/cc/winboll/studio/positions/activities/ShortcutActionActivity.java deleted file mode 100644 index f0618cc..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/activities/ShortcutActionActivity.java +++ /dev/null @@ -1,61 +0,0 @@ -package cc.winboll.studio.positions.activities; - -import android.app.Activity; -import android.content.Intent; -import android.os.Bundle; -import android.os.PersistableBundle; -import cc.winboll.studio.libappbase.ToastUtils; -import cc.winboll.studio.positions.R; -import cc.winboll.studio.positions.utils.APPPlusUtils; -import cc.winboll.studio.positions.utils.AppConfigsUtil; -import cc.winboll.studio.positions.AppLevel; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/15 13:45 - * @Describe 应用快捷方式活动类 - */ -public class ShortcutActionActivity extends Activity { - - public static final String TAG = "ShortcutActionActivity"; - - @Override - protected void onCreate(Bundle savedInstanceState) { - super.onCreate(savedInstanceState); - // 处理应用级别的切换请求 - handleSwitchRequest(); - finish(); - } - - - -// @Override -// public void onPostCreate(Bundle savedInstanceState, PersistableBundle persistentState) { -// super.onPostCreate(savedInstanceState, persistentState); -// finish(); -// } - -// @Override -// protected void onStart() { -// super.onStart(); -// } - - /** - * 处理应用图标快捷菜单的请求 - */ - private void handleSwitchRequest() { - Intent intent = getIntent(); - if (intent != null && "open_appplus".equals(intent.getDataString())) { - ToastUtils.show("已添加" + getString(R.string.app_name) + "附加组件"); - AppConfigsUtil.getInstance(getApplicationContext()).setAppLevel(AppLevel.LAOJUN); - APPPlusUtils.openAPPPlus(this); - //moveTaskToBack(true); - } - if (intent != null && "close_appplus".equals(intent.getDataString())) { - ToastUtils.show("已移除" + getString(R.string.app_name) + "附加组件"); - AppConfigsUtil.getInstance(getApplicationContext()).setAppLevel(AppLevel.WUKONG); - APPPlusUtils.closeAPPPlus(this); - //moveTaskToBack(true); - } - } -} diff --git a/positions/src/main/java/cc/winboll/studio/positions/activities/WinBoLLActivity.java b/positions/src/main/java/cc/winboll/studio/positions/activities/WinBoLLActivity.java index 8f40877..3cf13b7 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/activities/WinBoLLActivity.java +++ b/positions/src/main/java/cc/winboll/studio/positions/activities/WinBoLLActivity.java @@ -12,19 +12,11 @@ import androidx.appcompat.app.AppCompatActivity; import cc.winboll.studio.libaes.interfaces.IWinBoLLActivity; import cc.winboll.studio.libaes.utils.WinBoLLActivityManager; import cc.winboll.studio.libappbase.LogUtils; -import cc.winboll.studio.libappbase.ToastUtils; -import cc.winboll.studio.positions.App; -import cc.winboll.studio.positions.PointLevel; -import cc.winboll.studio.positions.R; -import cc.winboll.studio.positions.utils.ActivityAliasUtils; -import cc.winboll.studio.positions.utils.AppConfigsUtil; public class WinBoLLActivity extends AppCompatActivity implements IWinBoLLActivity { public static final String TAG = "WinBoLLActivity"; - public static volatile PointLevel _mPointLevel = PointLevel.WUKONG; - @Override public Activity getActivity() { return this; @@ -38,67 +30,18 @@ public class WinBoLLActivity extends AppCompatActivity implements IWinBoLLActivi @Override protected void onResume() { super.onResume(); - //ToastUtils.show("onResume"); - - // ActivityAliasUtils 工具使用示例 - // -// // 获取真实的目标组件名(即使通过 alias 启动,也能拿到 OriginalActivity) -// String realTargetName = ActivityAliasUtils.getRealTargetNameFromIntent(this); -// LogUtils.d("AliasActivity", "真实组件名:" + realTargetName); - // 获取真实的目标组件名(即使通过 alias 启动,也能拿到 OriginalActivity) -// String realTargetName = ActivityAliasUtils.getRealTargetNameFromIntent(this); -// LogUtils.d(TAG, "真实组件名:" + realTargetName); -// ToastUtils.show(realTargetName); -// // 判断某个组件是否为 alias -// String componentName = "com.winboll.app.AliasActivity"; -// boolean isAlias = ActivityAliasUtils.isActivityAlias(getApplicationContext(), componentName); -// LogUtils.d("判断结果", componentName + " 是否为 alias:" + isAlias); // true -// // 获取启动当前 Activity 的组件名(兼容 alias 场景) -// String launchComponent = ActivityAliasUtils.getLaunchComponentName(this); -// LogUtils.d("MainActivity", "启动组件名:" + launchComponent); - - /* - * 应用入口逻辑模块 - */ - // - // 检查当前活动的启动组件名,设置应用入口级别。 - String launchComponent = ActivityAliasUtils.getLaunchComponentName(this); - LogUtils.d("MainActivity", "启动组件名:" + launchComponent); - ToastUtils.show(launchComponent); - // 当前应用处于活动暂停的状态时,就检查应用的入口组件名称,设置应用入口级别。 - if (WinBoLLActivity._mPointLevel == PointLevel.DORAEMON) { - if (launchComponent.equals(App.COMPONENT_WUKONG)) { - getSupportActionBar().setTitle(getString(R.string.appplus_name)); - ToastUtils.show("WUKONG"); - _mPointLevel = PointLevel.WUKONG; - } else if (launchComponent.equals(App.COMPONENT_LAOJUN)) { - getSupportActionBar().setTitle(getString(R.string.app_name)); - ToastUtils.show("LAOJUN"); - _mPointLevel = PointLevel.LAOJUN; - } else { - // 如果是其他应用组件入口,就关闭活动 - finish(); - } - } - - /* - * 应用级别设置模块 - */ - // 读取并配置应用级别 - App._mAppLevel = AppConfigsUtil.getInstance(getApplicationContext()).getAppLevel(true); - LogUtils.d(TAG, String.format("onResume %s", getTag())); } @Override public boolean onOptionsItemSelected(MenuItem item) { /*if (item.getItemId() == R.id.item_log) { - WinBoLLActivityManager.getInstance().startLogActivity(this); - return true; - } else if (item.getItemId() == R.id.item_home) { - startActivity(new Intent(this, MainActivity.class)); - return true; - }*/ + WinBoLLActivityManager.getInstance().startLogActivity(this); + return true; + } else if (item.getItemId() == R.id.item_home) { + startActivity(new Intent(this, MainActivity.class)); + return true; + }*/ // 在switch语句中处理每个ID,并在处理完后返回true,未处理的情况返回false。 return super.onOptionsItemSelected(item); } diff --git a/positions/src/main/java/cc/winboll/studio/positions/models/AppConfigsModel.java b/positions/src/main/java/cc/winboll/studio/positions/models/AppConfigsModel.java index 032adb9..feb0aa0 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/models/AppConfigsModel.java +++ b/positions/src/main/java/cc/winboll/studio/positions/models/AppConfigsModel.java @@ -9,14 +9,12 @@ package cc.winboll.studio.positions.models; import android.util.JsonWriter; import android.util.JsonReader; import java.io.IOException; -import cc.winboll.studio.positions.AppLevel; public class AppConfigsModel extends BaseBean { public static final String TAG = "AppConfigsModel"; boolean isEnableMainService; - AppLevel appLevel; public AppConfigsModel(boolean isEnableMainService) { this.isEnableMainService = isEnableMainService; @@ -26,14 +24,6 @@ public class AppConfigsModel extends BaseBean { this.isEnableMainService = false; } - public void setAppLevel(AppLevel appLevel) { - this.appLevel = appLevel; - } - - public AppLevel getAppLevel() { - return appLevel; - } - public void setIsEnableMainService(boolean isEnableMainService) { this.isEnableMainService = isEnableMainService; } @@ -52,7 +42,6 @@ public class AppConfigsModel extends BaseBean { public void writeThisToJsonWriter(JsonWriter jsonWriter) throws IOException { super.writeThisToJsonWriter(jsonWriter); jsonWriter.name("isEnableDistanceRefreshService").value(isEnableMainService()); - jsonWriter.name("appLevel").value(getAppLevel().ordinal()); } // JSON反序列化(加载位置数据,校验字段) @@ -63,8 +52,6 @@ public class AppConfigsModel extends BaseBean { } else { if (name.equals("isEnableDistanceRefreshService")) { setIsEnableMainService(jsonReader.nextBoolean()); - } else if (name.equals("appLevel")) { - setAppLevel((AppLevel.values()[jsonReader.nextInt()])); } else { return false; } diff --git a/positions/src/main/java/cc/winboll/studio/positions/receivers/MotionStatusReceiver.java b/positions/src/main/java/cc/winboll/studio/positions/receivers/MotionStatusReceiver.java deleted file mode 100644 index 7ff5e8b..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/receivers/MotionStatusReceiver.java +++ /dev/null @@ -1,361 +0,0 @@ -package cc.winboll.studio.positions.receivers; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/10/28 19:07 - * @Describe MotionStatusReceiver - */ -import android.content.BroadcastReceiver; -import android.content.Context; -import android.content.Intent; -import android.content.pm.PackageManager; -import android.hardware.Sensor; -import android.hardware.SensorEvent; -import android.hardware.SensorEventListener; -import android.hardware.SensorManager; -import android.os.Build; -import android.os.Handler; -import android.os.Looper; -import android.text.TextUtils; -import cc.winboll.studio.libappbase.LogUtils; -import cc.winboll.studio.libappbase.ToastUtils; -import cc.winboll.studio.positions.services.MainService; -import cc.winboll.studio.positions.utils.ServiceUtil; - -/** - * 运动状态监听Receiver - * 功能:1.持续监听传感器(不关闭) 2.每5秒计算运动状态 3.按状态切换GPS模式(实时/30秒定时) - */ -public class MotionStatusReceiver extends BroadcastReceiver implements SensorEventListener { - public static final String TAG = "MotionStatusReceiver"; - - // 广播Action - public static final String ACTION_MOTION_STATUS_RECEIVER = "cc.winboll.studio.positions.receivers.MotionStatusReceiver"; - public static final String EXTRA_SENSORS_ENABLE = "EXTRA_SENSORS_ENABLE"; - // 传感器启动状态标志位 - boolean mIsSensorsEnable = false; - // 运动状态常量 - private static final int MOTION_STATUS_STATIC = 0; // 静止/低运动 - private static final int MOTION_STATUS_WALKING = 1; // 行走/高速运动 - // 配置参数(按需求调整) - private static final float ACCELEROMETER_THRESHOLD = 0.8f; // 加速度阈值 - private static final float GYROSCOPE_THRESHOLD = 0.5f; // 陀螺仪阈值 - private static final long STATUS_CALC_INTERVAL = 5000; // 运动状态计算间隔(5秒) - private static final long GPS_STATIC_INTERVAL = 30; // 静止时GPS间隔(30秒) - // 核心对象 - private volatile SensorManager mSensorManager; - private Sensor mAccelerometer; - private Sensor mGyroscope; - private volatile boolean mIsSensorListening = false; // 传感器是否持续监听 - private int mCurrentMotionStatus = MOTION_STATUS_STATIC; // 当前运动状态 - private Handler mMainHandler; // 主线程Handler(用于定时计算) - private Context mBroadcastContext; // 广播上下文 - // 传感器数据缓存(用于5秒内数据汇总,避免单次波动误判) - private float mAccelMax = 0f; // 5秒内加速度最大值 - private float mGyroMax = 0f; // 5秒内陀螺仪最大值 - - @Override - public void onReceive(Context context, Intent intent) { - - LogUtils.d(TAG, "===== 接收器启动:onReceive() 开始执行 ====="); - this.mBroadcastContext = context; - mMainHandler = new Handler(Looper.getMainLooper()); - - if (TextUtils.equals(intent.getAction(), ACTION_MOTION_STATUS_RECEIVER)) { - boolean isSettingEnable = intent.getBooleanExtra(EXTRA_SENSORS_ENABLE, false); - if (mIsSensorsEnable == false && isSettingEnable == true) { - mIsSensorsEnable = true; - // 1. 初始化传感器(必执行) - initSensors(); - - - if (mAccelerometer == null || mGyroscope == null) { - LogUtils.e(TAG, "设备缺少加速度/陀螺仪,无法持续监听"); - cleanResources(false); // 传感器不可用才清理 - return; - } - - // 2. 校验参数 - if (context == null || intent == null) { - LogUtils.d(TAG, "onReceive():无效参数,终止处理"); - cleanResources(false); - return; - } - LogUtils.d(TAG, "onReceive():接收到广播Action=" + intent.getAction()); - - // 3. 启动持续传感器监听(核心:不关闭,重复调用无影响) - startSensorListening(); - - // 4. 启动5秒定时计算运动状态(核心:持续触发状态判断) - startStatusCalcTimer(); - } - } - - - // 5. 处理外部广播触发(可选,保留外部控制能力) -// if (TextUtils.equals(intent.getAction(), ACTION_MOTION_STATUS_RECEIVER)) { -// int motionStatus = intent.getIntExtra(EXTRA_MOTION_STATUS, MOTION_STATUS_STATIC); -// String statusDesc = motionStatus == MOTION_STATUS_WALKING ? "高速运动" : "静止/低运动"; -// LogUtils.d(TAG, "外部广播触发,强制设置运动状态:" + statusDesc); -// mCurrentMotionStatus = motionStatus; -// handleMotionStatus(mCurrentMotionStatus); // 立即执行GPS切换 -// } - } - - /** - * 初始化传感器(持续监听,复用实例) - */ - private void initSensors() { - LogUtils.d(TAG, "initSensors():初始化传感器"); - if (mSensorManager != null || mBroadcastContext == null) return; - - mSensorManager = (SensorManager) mBroadcastContext.getSystemService(Context.SENSOR_SERVICE); - if (mSensorManager == null) { - LogUtils.e(TAG, "设备不支持传感器服务"); - return; - } - // 获取传感器实例(持续复用,不销毁) - mAccelerometer = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER); - mGyroscope = mSensorManager.getDefaultSensor(Sensor.TYPE_GYROSCOPE); - - LogUtils.d(TAG, "传感器初始化结果:加速度=" + (mAccelerometer != null) + ",陀螺仪=" + (mGyroscope != null)); - } - - /** - * 启动传感器持续监听(核心:不关闭,注册一次一直生效) - */ - private void startSensorListening() { - if (mSensorManager == null || mAccelerometer == null || mGyroscope == null) return; - - if (!mIsSensorListening) { - // 注册传感器监听(持续生效,直到服务销毁才注销) - mSensorManager.registerListener( - this, - mAccelerometer, - SensorManager.SENSOR_DELAY_NORMAL, // 正常延迟,平衡性能与精度 - mMainHandler - ); - mSensorManager.registerListener( - this, - mGyroscope, - SensorManager.SENSOR_DELAY_NORMAL, - mMainHandler - ); - mIsSensorListening = true; - LogUtils.d(TAG, "startSensorListening():传感器持续监听已启动(不关闭)"); - } - } - - /** - * 启动5秒定时计算运动状态(核心:周期性汇总传感器数据) - */ - private void startStatusCalcTimer() { - if (mMainHandler == null) return; - - // 移除旧任务(避免重复注册) - mMainHandler.removeCallbacks(mStatusCalcRunnable); - // 启动定时任务(每5秒执行一次) - mMainHandler.postDelayed(mStatusCalcRunnable, STATUS_CALC_INTERVAL); - LogUtils.d(TAG, "startStatusCalcTimer():5秒运动状态计算定时器已启动"); - } - - /** - * 运动状态计算任务(5秒执行一次) - */ - private final Runnable mStatusCalcRunnable = new Runnable() { - @Override - public void run() { - // 1. 基于5秒内缓存的最大传感器数据判断状态 - boolean isHighMotion = (mAccelMax > ACCELEROMETER_THRESHOLD) && (mGyroMax > GYROSCOPE_THRESHOLD); - int newMotionStatus = isHighMotion ? MOTION_STATUS_WALKING : MOTION_STATUS_STATIC; - - // 2. 状态变化时才处理(避免频繁切换GPS) - if (newMotionStatus != mCurrentMotionStatus) { - mCurrentMotionStatus = newMotionStatus; - String statusDesc = isHighMotion ? "高速运动" : "静止/低运动"; - LogUtils.d(TAG, "运动状态更新(5秒计算):" + statusDesc - + "(加速度最大值=" + mAccelMax + ",陀螺仪最大值=" + mGyroMax + ")"); - handleMotionStatus(newMotionStatus); // 切换GPS模式 - } else { - LogUtils.d(TAG, "运动状态无变化(5秒计算):" + (isHighMotion ? "高速运动" : "静止/低运动")); - } - - // 3. 重置传感器数据缓存,准备下一个5秒周期 - mAccelMax = 0f; - mGyroMax = 0f; - - // 4. 循环执行定时任务(核心:持续计算) - mMainHandler.postDelayed(this, STATUS_CALC_INTERVAL); - } - }; - - /** - * 传感器数据变化回调(核心:实时缓存最大数据) - */ - @Override - public void onSensorChanged(SensorEvent event) { - if (event == null) return; - - // 实时缓存5秒内的最大传感器数据(避免单次波动误判) - switch (event.sensor.getType()) { - case Sensor.TYPE_ACCELEROMETER: - float accelTotal = Math.abs(event.values[0]) + Math.abs(event.values[1]) + Math.abs(event.values[2]); - if (accelTotal > mAccelMax) mAccelMax = accelTotal; // 缓存最大值 - LogUtils.d(TAG, "加速度传感器实时数据:合值=" + accelTotal + "(当前5秒最大值=" + mAccelMax + ")"); - break; - case Sensor.TYPE_GYROSCOPE: - float gyroTotal = Math.abs(event.values[0]) + Math.abs(event.values[1]) + Math.abs(event.values[2]); - if (gyroTotal > mGyroMax) mGyroMax = gyroTotal; // 缓存最大值 - LogUtils.d(TAG, "陀螺仪实时数据:合值=" + gyroTotal + "(当前5秒最大值=" + mGyroMax + ")"); - break; - } - } - - /** - * 处理运动状态(核心:按状态切换GPS模式) - */ - private void handleMotionStatus(int motionStatus) { - LogUtils.d(TAG, "handleMotionStatus():开始处理运动状态,切换GPS模式"); - if (mBroadcastContext == null) { - LogUtils.w(TAG, "上下文为空,无法处理GPS"); - return; - } - - MainService mainService = getMainService(); - if (mainService == null) { - LogUtils.e(TAG, "MainService未启动,GPS控制失败"); - return; - } - - if (motionStatus == MOTION_STATUS_WALKING) { - // 高速运动:启动GPS实时更新(2秒/1米) - handleHighMotionGPS(mainService); - } else { - // 静止/低运动:启动GPS30秒定时更新 - handleStaticGPS(mainService); - } - } - - /** - * 高速运动GPS处理:实时更新 - */ - private void handleHighMotionGPS(MainService mainService) { - // 动态权限判断(Android 6.0+) - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M && - mBroadcastContext.checkSelfPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) - != PackageManager.PERMISSION_GRANTED) { - sendPermissionRequestBroadcast(); - return; - } - - // 启动实时GPS(已启动则不重复操作) - if (!mainService.isGpsListening()) { - mainService.startGpsLocation(); // 实时更新(2秒/1米) - mainService.stopGpsStaticTimer(); // 停止定时GPS - LogUtils.d(TAG, "高速运动:已启动GPS实时更新"); - } - } - - /** - * 静止/低运动GPS处理:30秒定时更新 - */ - private void handleStaticGPS(MainService mainService) { - // 停止实时GPS(已停止则不重复操作) - if (mainService.isGpsListening()) { - mainService.stopGpsLocation(); // 停止实时更新 - LogUtils.d(TAG, "静止/低运动:已停止GPS实时更新"); - } - - // 启动30秒定时GPS(已启动则不重复操作) - mainService.startGpsStaticTimer(GPS_STATIC_INTERVAL); // 30秒一次 - LogUtils.d(TAG, "静止/低运动:已启动GPS30秒定时更新"); - } - - /** - * 获取MainService实例(复用逻辑) - */ - private MainService getMainService() { - if (mBroadcastContext == null) return null; - - // 优先获取单例 - MainService singleton = MainService.getInstance(mBroadcastContext); - if (singleton != null && singleton.isServiceRunning()) { - return singleton; - } - - // 启动服务并重试 - if (!ServiceUtil.isServiceAlive(mBroadcastContext, MainService.class.getName())) { - mBroadcastContext.startService(new Intent(mBroadcastContext, MainService.class)); - try { - Thread.sleep(500); // 等待服务启动 - } catch (InterruptedException e) { - Thread.currentThread().interrupt(); - } - } - return MainService.getInstance(mBroadcastContext); - } - - /** - * 发送GPS权限申请广播(Receiver无法直接申请) - */ - private void sendPermissionRequestBroadcast() { - Intent permissionIntent = new Intent("cc.winboll.studio.positions.ACTION_REQUEST_GPS_PERMISSION"); - permissionIntent.addFlags(Intent.FLAG_INCLUDE_STOPPED_PACKAGES); - mBroadcastContext.sendBroadcast(permissionIntent); - LogUtils.d(TAG, "GPS权限缺失,已发送申请广播"); - } - - /** - * 资源清理(核心:传感器不关闭,仅清理Handler和上下文) - * @param isForceStopSensor 是否强制停止传感器(仅服务销毁时传true) - */ - private void cleanResources(boolean isForceStopSensor) { - // 1. 停止定时计算任务 - if (mMainHandler != null) { - mMainHandler.removeCallbacksAndMessages(null); - mMainHandler = null; - LogUtils.d(TAG, "cleanResources():已停止运动状态计算定时器"); - } - - // 2. 强制停止传感器(仅当外部触发销毁时执行,正常情况不关闭) - if (isForceStopSensor && mSensorManager != null && mIsSensorListening) { - mSensorManager.unregisterListener(this); - mIsSensorListening = false; - LogUtils.d(TAG, "cleanResources():已强制停止传感器监听"); - } - - // 3. 置空上下文(避免内存泄漏) - mBroadcastContext = null; - } - - /** - * 传感器精度变化回调(日志监控) - */ - @Override - public void onAccuracyChanged(Sensor sensor, int accuracy) { - String sensorType = sensor.getType() == Sensor.TYPE_ACCELEROMETER ? "加速度" : "陀螺仪"; - String accuracyDesc = getAccuracyDesc(accuracy); - LogUtils.d(TAG, sensorType + "传感器精度变化:" + accuracyDesc); - } - - /** - * 传感器精度描述转换 - */ - private String getAccuracyDesc(int accuracy) { - switch (accuracy) { - case SensorManager.SENSOR_STATUS_ACCURACY_HIGH: return "高"; - case SensorManager.SENSOR_STATUS_ACCURACY_MEDIUM: return "中"; - case SensorManager.SENSOR_STATUS_ACCURACY_LOW: return "低"; - case SensorManager.SENSOR_STATUS_UNRELIABLE: return "不可靠"; - default: return "未知"; - } - } - - /** - * 补充:Receiver销毁时强制清理(需在MainService注销时调用) - */ - public void forceCleanResources() { - cleanResources(true); // 强制停止传感器 - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/services/MainService.java b/positions/src/main/java/cc/winboll/studio/positions/services/MainService.java index c68884f..59b4bc0 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/services/MainService.java +++ b/positions/src/main/java/cc/winboll/studio/positions/services/MainService.java @@ -3,7 +3,7 @@ package cc.winboll.studio.positions.services; /** * @Author ZhanGSKen * @Date 2024/07/19 14:30:57 - * @Describe 应用主要服务组件类(适配运动状态GPS控制+静止10秒切换) + * @Describe 应用主要服务组件类 */ import android.app.Service; import android.content.ComponentName; @@ -35,216 +35,121 @@ import java.util.Iterator; import java.util.Set; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; -import java.util.concurrent.TimeUnit; -import cc.winboll.studio.positions.utils.LocalMotionDetector; +import java.util.concurrent.TimeUnit; // 新增:定时器时间单位依赖 -public class MainService extends Service implements LocalMotionDetector.MotionStatusCallback { +public class MainService extends Service { public static final String TAG = "MainService"; - public static final String EXTRA_IS_SETTING_TO_ENABLE = "EXTRA_IS_SETTING_TO_ENABLE"; - // ---------------------- 核心:LocalMotionDetector 实例(唯一运动状态检测工具) ---------------------- - private LocalMotionDetector mMotionDetector; // 本地运动状态检测器 - private static final long GPS_STATIC_INTERVAL = 30; // 静止时GPS间隔(30秒) + public static final String EXTRA_IS_SETTING_TO_ENABLE = "EXTRA_IS_SETTING_TO_ENABLE"; - // ---------------------- 定时器相关变量 ---------------------- - private ScheduledExecutorService gpsStaticTimer; // 静止时GPS定时任务 - private volatile boolean isGpsListening = false; // GPS是否持续监听 - private static final long GPS_STATIC_DURATION = 3; // 单次GPS获取超时(秒) + // ---------------------- 新增:定时器相关变量 ---------------------- + private ScheduledExecutorService taskCheckTimer; // 任务校验定时器 + private static final long TASK_CHECK_INTERVAL = 1; // 定时间隔(1分钟) + private static final long TASK_CHECK_INIT_DELAY = 1; // 初始延迟(1分钟:立即启动) - // ---------------------- 静止延迟切换配置(核心功能) ---------------------- - private static final long STATIC_DELAY_MS = 10 * 1000; // 静止10秒后切换 - private Handler staticDelayHandler; // 延迟执行切换逻辑的Handler - private volatile boolean isDelaySwitching = false; // 标记是否正在等待延迟切换 - - // GPS监听接口 + // GPS监听接口(Java 7 标准接口定义,无Lambda依赖) public interface GpsUpdateListener { void onGpsPositionUpdated(PositionModel currentGpsPos); void onGpsStatusChanged(String status); } - // 任务更新监听接口 + // 任务更新监听接口(Java 7 风格,供Adapter监听任务变化) public interface TaskUpdateListener { void onTaskUpdated(); } - // 监听管理 + // 监听管理(弱引用+线程安全集合,适配Java 7,避免内存泄漏+并发异常) private final Set> mGpsListeners = new HashSet>(); private final Set> mTaskListeners = new HashSet>(); - private final Object mListenerLock = new Object(); + private final Object mListenerLock = new Object(); // 监听操作锁,保证线程安全 - // 原有核心变量 - private LocalBinder mLocalBinder; + // 原有核心变量(Java 7 显式初始化,无Java 8+语法) + private LocalBinder mLocalBinder; //持有 LocalBinder 实例(用于暴露服务) private LocationManager mLocationManager; private LocationListener mGpsLocationListener; - private static final long GPS_UPDATE_INTERVAL = 2000; // 实时GPS更新间隔:2秒 - private static final float GPS_UPDATE_DISTANCE = 1; // 实时GPS更新距离:1米 - private boolean isGpsEnabled = false; - private boolean isGpsPermissionGranted = false; + private static final long GPS_UPDATE_INTERVAL = 2000; // GPS更新间隔:2秒 + private static final float GPS_UPDATE_DISTANCE = 1; // GPS更新距离阈值:1米 + private boolean isGpsEnabled = false; // GPS是否启用标记 + private boolean isGpsPermissionGranted = false; // 定位权限是否授予标记 - // 数据存储集合 - private final ArrayList mPositionList = new ArrayList(); - private final ArrayList mAllTasks = new ArrayList(); - private static PositionModel _mCurrentGpsPosition; + // 数据存储集合(Java 7 基础集合,避免Stream/forEach等Java 8+特性) + private final ArrayList mPositionList = new ArrayList(); // 位置数据列表 + private final ArrayList mAllTasks = new ArrayList();// 任务数据列表 + private static PositionModel _mCurrentGpsPosition; // 当前GPS定位数据 - // 服务相关变量 + // 服务相关变量(Java 7 显式声明,保持原逻辑) MyServiceConnection mMyServiceConnection; - volatile static boolean _mIsServiceRunning; + volatile static boolean _mIsServiceRunning; // 服务运行状态(volatile保证可见性) AppConfigsUtil mAppConfigsUtil; - private ScheduledExecutorService distanceExecutor = Executors.newSingleThreadScheduledExecutor(); - private final Set mVisiblePositionIds = new HashSet(); + private ScheduledExecutorService distanceExecutor = Executors.newSingleThreadScheduledExecutor(); // 单线程池处理距离计算 + private final Set mVisiblePositionIds = new HashSet(); // 可见位置ID集合 - // 单例+应用上下文 + // 单例+应用上下文(Java 7 静态变量,保证服务实例唯一+上下文安全) private static volatile MainService sInstance; private static Context sAppContext; // ========================================================================= - // 核心:运动状态回调(仅通过LocalMotionDetector触发,低运动归类为静止) + // 新增:定时器初始化方法(创建单线程定时器,每1分钟调用任务校验) // ========================================================================= - @Override - public void onMotionStatusChanged(boolean isWalking, String statusDesc) { - LogUtils.d(TAG, "运动状态回调:" + statusDesc + "(isWalking=" + isWalking + ")"); - updateNotificationGpsStatus("当前状态:" + statusDesc); - - // 仅行走状态切实时GPS,非行走(静止+低运动)统一按静止处理 - if (isWalking) { - // 行走状态:立即切回实时GPS(取消未执行的延迟任务) - if (staticDelayHandler != null) { - staticDelayHandler.removeCallbacksAndMessages(null); // 清空延迟任务 - isDelaySwitching = false; // 重置标记 - } - stopGpsStaticTimer(); // 停止定时GPS - startGpsLocation(); // 启动实时GPS - LogUtils.d(TAG, "行走状态:已切换为实时GPS(2秒/1米)"); - - } else { - // 非行走状态:延迟10秒切换为定时GPS - if (isGpsListening && !isDelaySwitching) { // 仅实时GPS运行且无延迟任务时触发 - isDelaySwitching = true; - staticDelayHandler.postDelayed(new Runnable() { - @Override - public void run() { - stopGpsLocation(); // 停止实时GPS - startGpsStaticTimer(GPS_STATIC_INTERVAL); // 启动30秒定时GPS - LogUtils.d(TAG, "静止/低运动10秒:已切换为30秒定时GPS"); - isDelaySwitching = false; // 重置标记 - } - }, STATIC_DELAY_MS); - LogUtils.d(TAG, "检测到静止/低运动:将在10秒后切换为定时GPS(若持续非行走)"); - } - LogUtils.d(TAG, "静止/低运动状态:等待延迟切换GPS"); + private void initTaskCheckTimer() { + // 先销毁旧定时器(避免重复创建导致多线程问题) + if (taskCheckTimer != null && !taskCheckTimer.isShutdown()) { + taskCheckTimer.shutdown(); } - } - - // ========================================================================= - // 静止GPS定时任务方法(适配Java7) - // ========================================================================= - // 启动静止时GPS定时获取(间隔:秒) - public void startGpsStaticTimer(long intervalSeconds) { - stopGpsStaticTimer(); - - gpsStaticTimer = Executors.newSingleThreadScheduledExecutor(); - gpsStaticTimer.scheduleAtFixedRate(new Runnable() { + // 创建单线程定时器(确保任务串行执行,避免并发异常) + taskCheckTimer = Executors.newSingleThreadScheduledExecutor(); + // 定时任务:初始延迟1分钟,每1分钟执行一次 + taskCheckTimer.scheduleAtFixedRate(new Runnable() { @Override public void run() { - LogUtils.d(TAG, "静止时GPS定时任务触发:开始单次GPS获取"); - startSingleGpsRetrieve(); + LogUtils.d(TAG, "定时任务触发:开始校验任务(间隔1分钟)"); + // 调用任务校验核心方法(与GPS位置变化时逻辑一致) + DistanceCalculatorUtil.getInstance(MainService.this).checkAllTaskTriggerCondition(MainService._mCurrentGpsPosition); } - }, 0, intervalSeconds, TimeUnit.SECONDS); + }, TASK_CHECK_INIT_DELAY, TASK_CHECK_INTERVAL, TimeUnit.MINUTES); - LogUtils.d(TAG, "静止时GPS定时任务已启动(间隔:" + intervalSeconds + "秒)"); + LogUtils.d(TAG, "任务校验定时器已启动(间隔:" + TASK_CHECK_INTERVAL + "分钟)"); } - // 停止静止时GPS定时任务 - public void stopGpsStaticTimer() { - if (gpsStaticTimer != null && !gpsStaticTimer.isShutdown()) { - gpsStaticTimer.shutdown(); + // ========================================================================= + // 新增:定时器销毁方法(服务销毁时调用,避免内存泄漏) + // ========================================================================= + private void destroyTaskCheckTimer() { + if (taskCheckTimer != null && !taskCheckTimer.isShutdown()) { + taskCheckTimer.shutdown(); // 优雅关闭:等待已提交任务执行完成 try { - if (!gpsStaticTimer.awaitTermination(1, TimeUnit.SECONDS)) { - gpsStaticTimer.shutdownNow(); + // 等待1秒,若未终止则强制关闭 + if (!taskCheckTimer.awaitTermination(1, TimeUnit.SECONDS)) { + taskCheckTimer.shutdownNow(); // 强制终止未完成任务 } } catch (InterruptedException e) { - gpsStaticTimer.shutdownNow(); - Thread.currentThread().interrupt(); + taskCheckTimer.shutdownNow(); // 捕获中断异常,强制关闭 + Thread.currentThread().interrupt(); // 恢复线程中断状态 } finally { - gpsStaticTimer = null; - LogUtils.d(TAG, "静止时GPS定时任务已销毁"); + taskCheckTimer = null; // 置空,避免重复操作 + LogUtils.d(TAG, "任务校验定时器已销毁"); } } } - // 单次GPS获取(获取后关闭监听) - private void startSingleGpsRetrieve() { - if (!checkGpsReady()) { - LogUtils.w(TAG, "单次GPS获取:GPS未就绪,跳过"); - return; - } - - try { - mLocationManager.requestSingleUpdate( - LocationManager.GPS_PROVIDER, - new LocationListener() { - @Override - public void onLocationChanged(Location location) { - if (location != null) { - PositionModel gpsPos = new PositionModel(); - gpsPos.setLatitude(location.getLatitude()); - gpsPos.setLongitude(location.getLongitude()); - gpsPos.setPositionId("CURRENT_GPS_POS"); - gpsPos.setMemo("静止时单次GPS位置"); - syncCurrentGpsPosition(gpsPos); - LogUtils.d(TAG, "单次GPS获取成功:纬度=" + location.getLatitude()); - } - mLocationManager.removeUpdates(this); - } - - @Override - public void onStatusChanged(String provider, int status, Bundle extras) {} - - @Override - public void onProviderEnabled(String provider) {} - - @Override - public void onProviderDisabled(String provider) { - mLocationManager.removeUpdates(this); - } - }, - Looper.getMainLooper() - ); - - // 超时处理 - new Handler(Looper.getMainLooper()).postDelayed(new Runnable() { - @Override - public void run() { - mLocationManager.removeUpdates(new LocationListener() { - @Override public void onLocationChanged(Location location) {} - @Override public void onStatusChanged(String provider, int status, Bundle extras) {} - @Override public void onProviderEnabled(String provider) {} - @Override public void onProviderDisabled(String provider) {} - }); - LogUtils.d(TAG, "单次GPS获取超时(" + GPS_STATIC_DURATION + "秒)"); - } - }, GPS_STATIC_DURATION * 1000); - - } catch (SecurityException e) { - LogUtils.e(TAG, "单次GPS获取失败(权限异常):" + e.getMessage()); - } catch (Exception e) { - LogUtils.e(TAG, "单次GPS获取失败:" + e.getMessage()); - } - } - - // ========================================================================= - // 任务操作核心接口 + // 任务操作核心接口(Java 7 实现,全迭代器遍历,无ConcurrentModificationException) // ========================================================================= + /** + * 新增任务(Adapter调用,通过MainService统一管理任务,保证数据一致性) + * @param newTask 待新增的任务模型 + */ public void addTask(PositionTaskModel newTask) { + // 参数校验(Java 7 基础判断,无Optional等Java 8+特性) if (newTask == null || TextUtils.isEmpty(newTask.getPositionId())) { LogUtils.w(TAG, "addPositionTask:任务为空或未绑定位置ID,新增失败"); return; } + // 任务去重(Java 7 迭代器遍历,避免增强for循环删除/新增导致的并发异常) boolean isDuplicate = false; Iterator taskIter = mAllTasks.iterator(); while (taskIter.hasNext()) { @@ -259,18 +164,25 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt return; } + // 新增任务+持久化+通知刷新(全Java 7 语法) mAllTasks.add(newTask); saveAllTasks(); - notifyTaskUpdated(); + notifyTaskUpdated(); // 通知所有监听者(如Adapter)任务已更新 LogUtils.d(TAG, "addPositionTask:成功(位置ID=" + newTask.getPositionId() + ",任务ID=" + newTask.getTaskId() + ")"); } + /** + * 获取指定位置的所有任务(Adapter显示任务数量用,数据来源唯一) + * @param positionId 位置ID + * @return 该位置绑定的所有任务(返回新列表,避免外部修改原数据) + */ public ArrayList getTasksByPositionId(String positionId) { ArrayList posTasks = new ArrayList(); if (TextUtils.isEmpty(positionId) || mAllTasks.isEmpty()) { return posTasks; } + // 筛选任务(Java 7 迭代器遍历,安全筛选) Iterator taskIter = mAllTasks.iterator(); while (taskIter.hasNext()) { PositionTaskModel task = taskIter.next(); @@ -281,44 +193,55 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt return posTasks; } + /** + * 获取所有任务(Adapter全量刷新用,返回拷贝避免原数据被外部修改) + * @return 所有任务的拷贝列表 + */ public ArrayList getAllTasks() { - return mAllTasks; + return mAllTasks; // Java 7 集合拷贝方式 } - public void updateTask(PositionTaskModel updatedTask) { + public void updateTask(PositionTaskModel updatedTask) { if (updatedTask == null || updatedTask.getTaskId() == null) return; for (int i = 0; i < mAllTasks.size(); i++) { PositionTaskModel task = mAllTasks.get(i); if (updatedTask.getTaskId().equals(task.getTaskId())) { - mAllTasks.set(i, updatedTask); + mAllTasks.set(i, updatedTask); // 替换为更新后的任务 break; } - } - saveAllTasks(); + } + saveAllTasks(); // 持久化更新后的数据 } + // 4. 仅更新任务启用状态(优化性能,避免全量字段更新) public void updateTaskStatus(PositionTaskModel task) { if (task == null || task.getTaskId() == null) return; for (PositionTaskModel item : mAllTasks) { if (task.getTaskId().equals(item.getTaskId())) { - item.setIsEnable(task.isEnable()); + item.setIsEnable(task.isEnable()); // 只更新启用状态字段 break; } } - saveAllTasks(); + saveAllTasks(); // 持久化状态变更 } + + /** + * 删除任务(Adapter调用,通过迭代器安全删除,避免并发异常) + * @param taskId 待删除任务的ID + */ public void deleteTask(String taskId) { if (TextUtils.isEmpty(taskId) || mAllTasks.isEmpty()) { LogUtils.w(TAG, "deletePositionTask:任务ID为空或列表为空,删除失败"); return; } + // 迭代器删除(Java 7 唯一安全删除集合元素的方式) Iterator taskIter = mAllTasks.iterator(); while (taskIter.hasNext()) { PositionTaskModel task = taskIter.next(); if (taskId.equals(task.getTaskId())) { - taskIter.remove(); + taskIter.remove(); // 迭代器安全删除,无ConcurrentModificationException saveAllTasks(); notifyTaskUpdated(); LogUtils.d(TAG, "deletePositionTask:成功(任务ID=" + taskId + ")"); @@ -327,16 +250,24 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } } + /** + * 注册任务更新监听(Java 7 弱引用管理,避免内存泄漏) + * @param listener 任务更新监听者(如Adapter) + */ public void registerTaskUpdateListener(TaskUpdateListener listener) { if (listener == null) { LogUtils.w(TAG, "registerTaskUpdateListener:监听者为空,跳过"); return; } - synchronized (mListenerLock) { + synchronized (mListenerLock) { // 加锁保证多线程注册安全 mTaskListeners.add(new WeakReference(listener)); } } + /** + * 反注册任务更新监听(Java 7 迭代器清理,避免内存泄漏) + * @param listener 待反注册的监听者 + */ public void unregisterTaskUpdateListener(TaskUpdateListener listener) { if (listener == null) { LogUtils.w(TAG, "unregisterTaskUpdateListener:监听者为空,跳过"); @@ -346,6 +277,7 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt Iterator> iter = mTaskListeners.iterator(); while (iter.hasNext()) { WeakReference ref = iter.next(); + // 清理目标监听者或已被回收的弱引用 if (ref.get() == listener || ref.get() == null) { iter.remove(); } @@ -353,12 +285,16 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } } + /** + * 通知所有任务监听者更新(Java 7 匿名内部类实现主线程回调,无Lambda) + */ private void notifyTaskUpdated() { synchronized (mListenerLock) { Iterator> iter = mTaskListeners.iterator(); while (iter.hasNext()) { final WeakReference ref = iter.next(); if (ref.get() != null) { + // 判断是否在主线程,不在则切换(Java 7 匿名Runnable,无Lambda) if (Looper.myLooper() == Looper.getMainLooper()) { ref.get().onTaskUpdated(); } else { @@ -370,7 +306,7 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt }); } } else { - iter.remove(); + iter.remove(); // 清理已回收的弱引用,避免内存泄漏 } } } @@ -378,14 +314,19 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt // ========================================================================= - // 服务基础方法(初始化LocalMotionDetector,删除Receiver相关逻辑) + // 原有基础方法(Java 7 语法调整:移除所有Lambda/方法引用,用匿名内部类替代) // ========================================================================= + /** + * 获取服务单例(Java 7 静态同步方法,保证线程安全) + * @param context 上下文 + * @return MainService实例(未绑定成功时返回null) + */ public static synchronized MainService getInstance(Context context) { if (sInstance == null) { - if (AppConfigsUtil.getInstance(context).isEnableMainService(true)) { - Intent intent = new Intent(context.getApplicationContext(), MainService.class); - context.getApplicationContext().startService(intent); - } + if (AppConfigsUtil.getInstance(context).isEnableMainService(true)) { + Intent intent = new Intent(context.getApplicationContext(), MainService.class); + context.getApplicationContext().startService(intent); + } return null; } if (sAppContext == null) { @@ -394,11 +335,18 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt return sInstance; } + /** + * 服务绑定回调(Java 7 基础实现,无默认方法等Java 8+特性) + */ @Override - public IBinder onBind(Intent intent) { - return mLocalBinder; - } + public IBinder onBind(Intent intent) { + // 返回 LocalBinder,使Activity能通过Binder获取MainService实例 + return mLocalBinder; + } + /** + * 服务创建回调(初始化单例、上下文、配置、服务连接等) + */ @Override public void onCreate() { LogUtils.d(TAG, "onCreate"); @@ -406,118 +354,129 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt sInstance = this; sAppContext = getApplicationContext(); - mLocalBinder = new LocalBinder(this); + // 初始化 LocalBinder(关键:将MainService实例传入Binder) + mLocalBinder = new LocalBinder(this); + _mIsServiceRunning = false; mAppConfigsUtil = AppConfigsUtil.getInstance(this); + // 初始化服务连接(Java 7 显式判断,无Optional) if (mMyServiceConnection == null) { mMyServiceConnection = new MyServiceConnection(); } - // 初始化:LocalMotionDetector+延迟切换Handler(Java7兼容) - mMotionDetector = LocalMotionDetector.getInstance(); - staticDelayHandler = new Handler(Looper.getMainLooper()); - - if (mAppConfigsUtil.isEnableMainService(true)) { - run(); - } + if (mAppConfigsUtil.isEnableMainService(true)) { + run(); // 启动服务核心逻辑 + } } - // 服务核心运行方法(删除Receiver注册逻辑) + /** + * 服务核心逻辑(启动前台服务、初始化GPS、加载数据等) + * 【关键修改】新增定时器初始化,每1分钟调用任务校验 + */ public void run() { if (mAppConfigsUtil.isEnableMainService(true)) { if (!_mIsServiceRunning) { _mIsServiceRunning = true; - wakeupAndBindAssistant(); + wakeupAndBindAssistant(); // 唤醒并绑定辅助服务 - // 启动前台服务 + // 启动前台服务(Java 7 显式调用,无方法引用) String initialStatus = "[ Positions ] is in Service."; - NotificationUtil.createForegroundServiceNotification(this, initialStatus); + NotificationUtil.createForegroundServiceNotification(this, initialStatus); startForeground(NotificationUtil.FOREGROUND_SERVICE_NOTIFICATION_ID, NotificationUtil.createForegroundServiceNotification(this, initialStatus)); - // 初始化LocationManager + // 初始化GPS相关(Java 7 基础API调用) mLocationManager = (LocationManager) sInstance.getApplicationContext().getSystemService(Context.LOCATION_SERVICE); + initGpsLocationListener(); + startGpsLocation(); - // 加载本地数据 + // 加载本地数据(Java 7 静态方法调用,无方法引用) PositionModel.loadBeanList(MainService.this, mPositionList, PositionModel.class); PositionTaskModel.loadBeanList(MainService.this, mAllTasks, PositionTaskModel.class); - // 核心:启动LocalMotionDetector监测(唯一运动状态来源) - mMotionDetector.startDetection(this, this); - - // 提示与日志 + // 提示与日志(Java 7 基础调用) ToastUtils.show(initialStatus); LogUtils.i(TAG, initialStatus); + + // ---------------------- 关键新增:启动任务校验定时器 ---------------------- + //checkAllTaskTriggerCondition(); + initTaskCheckTimer(); } } } + /** + * 获取服务运行状态 + * @return true=运行中,false=未运行 + */ public boolean isServiceRunning() { return _mIsServiceRunning; } + /** + * 服务销毁回调(清理资源、停止GPS、清空数据、反注册监听等) + * 【关键修改】新增定时器销毁,避免内存泄漏 + */ @Override public void onDestroy() { super.onDestroy(); sInstance = null; - // 核心:停止LocalMotionDetector(避免传感器内存泄漏) - if (mMotionDetector != null) { - mMotionDetector.stopDetection(); - LogUtils.d(TAG, "LocalMotionDetector 已停止检测"); - } - - // 清理延迟Handler任务(避免内存泄漏) - if (staticDelayHandler != null) { - staticDelayHandler.removeCallbacksAndMessages(null); - staticDelayHandler = null; - } - - // 原有清理逻辑(已删除Receiver注销代码) + // 清理资源(Java 7 顺序调用,无Stream等特性) stopGpsLocation(); - stopGpsStaticTimer(); clearAllData(); stopForeground(true); + // 清理所有监听者(Java 7 加锁+清空,避免内存泄漏) synchronized (mListenerLock) { mGpsListeners.clear(); mTaskListeners.clear(); } + // ---------------------- 关键新增:销毁任务校验定时器 ---------------------- + destroyTaskCheckTimer(); + // 销毁距离计算线程池(原有逻辑,补充确保线程安全) if (distanceExecutor != null && !distanceExecutor.isShutdown()) { distanceExecutor.shutdown(); } + // 重置状态变量 _mIsServiceRunning = false; isGpsEnabled = false; - isGpsListening = false; - isDelaySwitching = false; mLocationManager = null; - LogUtils.d(TAG, "MainService 销毁完成,所有资源已清理"); } // ========================================================================= - // 位置操作方法 + // 位置操作方法(Java 7 语法,全迭代器/基础循环,无Java 8+特性) // ========================================================================= + /** + * 获取所有位置数据(返回原列表,供外部读取) + * @return 位置列表 + */ public ArrayList getPositionList() { return mPositionList; } + /** + * 获取当前GPS位置 + * @return 当前GPS定位模型(未获取时返回null) + */ public PositionModel getCurrentGpsPosition() { return _mCurrentGpsPosition; } - public boolean isGpsListening() { - return isGpsListening; - } - + /** + * 删除指定位置(Java 7 迭代器安全删除) + * @param targetPosId 待删除位置的ID + */ public void removePosition(String targetPosId) { if (TextUtils.isEmpty(targetPosId) || mPositionList.isEmpty()) { LogUtils.w(TAG, "removePosition:参数无效"); return; } + // 迭代器遍历删除(Java 7 安全方式) Iterator iter = mPositionList.iterator(); while (iter.hasNext()) { PositionModel pos = iter.next(); @@ -529,11 +488,16 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } } + /** + * 更新位置数据(Java 7 基础for循环,无Stream筛选) + * @param updatedPos 更新后的位置模型 + */ public void updatePosition(PositionModel updatedPos) { if (updatedPos == null || TextUtils.isEmpty(updatedPos.getPositionId()) || mPositionList.isEmpty()) { LogUtils.w(TAG, "updatePosition:参数无效"); return; } + // 基础for循环查找并更新(Java 7 标准写法) for (int i = 0; i < mPositionList.size(); i++) { PositionModel oldPos = mPositionList.get(i); if (updatedPos.getPositionId().equals(oldPos.getPositionId())) { @@ -544,22 +508,32 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } } + /** + * 同步所有位置任务(全量替换,用于批量更新) + * @param newTaskList 新的任务列表 + */ public void syncAllPositionTasks(ArrayList newTaskList) { if (newTaskList == null) { LogUtils.w(TAG, "syncAllPositionTasks:新列表为空"); return; } + // 全量替换+持久化+通知(Java 7 基础集合操作) mAllTasks.clear(); mAllTasks.addAll(newTaskList); saveAllTasks(); notifyTaskUpdated(); } + /** + * 新增位置(Java 7 增强for循环去重,无Stream) + * @param newPos 待新增的位置模型 + */ public void addPosition(PositionModel newPos) { if (newPos == null) { LogUtils.w(TAG, "addPosition:位置为空"); return; } + // 位置去重(Java 7 增强for循环,无Stream.filter) boolean isDuplicate = false; for (PositionModel pos : mPositionList) { if (newPos.getPositionId().equals(pos.getPositionId())) { @@ -573,16 +547,25 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } } + /** + * 持久化位置数据(Java 7 静态方法调用,保持原逻辑) + */ void savePositionList() { LogUtils.d(TAG, String.format("savePositionList : size=%d", mPositionList.size())); PositionModel.saveBeanList(MainService.this, mPositionList, PositionModel.class); } + /** + * 持久化任务数据(Java 7 静态方法调用,保持原逻辑) + */ public void saveAllTasks() { LogUtils.d(TAG, String.format("saveTaskList : size=%d", mAllTasks.size())); PositionTaskModel.saveBeanList(MainService.this, mAllTasks, PositionTaskModel.class); } + /** + * 清空所有数据(位置+任务+GPS缓存,Java 7 集合clear方法) + */ public void clearAllData() { mPositionList.clear(); mAllTasks.clear(); @@ -590,6 +573,10 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt LogUtils.d(TAG, "clearAllData:已清空所有数据"); } + /** + * 同步当前GPS位置(更新缓存+通知监听者+同步通知栏,全Java 7 语法) + * @param position 最新GPS位置模型 + */ public void syncCurrentGpsPosition(PositionModel position) { if (position == null) { LogUtils.w(TAG, "syncCurrentGpsPosition:位置为空"); @@ -599,21 +586,27 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt LogUtils.d(TAG, "syncCurrentGpsPosition:成功(纬度=" + position.getLatitude() + ",经度=" + position.getLongitude() + ")"); notifyAllGpsListeners(position); + // 服务运行中才同步通知栏状态 if (_mIsServiceRunning) { syncGpsStatusToNotification(); } } + /** + * 同步GPS状态到前台通知(Java 7 匿名Runnable切换主线程,无Lambda) + */ private void syncGpsStatusToNotification() { if (!_mIsServiceRunning || _mCurrentGpsPosition == null) { return; } + // 格式化通知内容(Java 7 String.format,无String.join等Java 8+方法) final String gpsStatus = String.format( - "GPS位置:北纬%.4f° 东经%.4f° | 可见位置:%d个", - _mCurrentGpsPosition.getLatitude(), - _mCurrentGpsPosition.getLongitude(), - mVisiblePositionIds.size() + "GPS位置:北纬%.4f° 东经%.4f° | 可见位置:%d个", + _mCurrentGpsPosition.getLatitude(), + _mCurrentGpsPosition.getLongitude(), + mVisiblePositionIds.size() ); + // 主线程判断+切换(Java 7 匿名内部类) if (Looper.myLooper() == Looper.getMainLooper()) { NotificationUtil.updateForegroundServiceStatus(this, gpsStatus); } else { @@ -627,35 +620,54 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } + + // ========================================================================= - // 服务生命周期+辅助服务相关 + // 服务生命周期+辅助服务相关(Java 7 语法,无Lambda/方法引用) // ========================================================================= + /** + * 服务启动命令(每次startService调用时触发,重启服务核心逻辑) + */ @Override public int onStartCommand(Intent intent, int flags, int startId) { - boolean isSettingToEnable = false; - if (intent != null) { - isSettingToEnable = intent.getBooleanExtra(EXTRA_IS_SETTING_TO_ENABLE, false); - if (isSettingToEnable) { - run(); - } - } + boolean isSettingToEnable = false; + if (intent != null) { + isSettingToEnable = intent.getBooleanExtra(EXTRA_IS_SETTING_TO_ENABLE, false); + if (isSettingToEnable) { + run(); // 重启服务核心逻辑(保证服务启动后进入运行状态) + } + } + + // 如果被设置为自启动就返回START_STICKY:服务被异常杀死后,系统会尝试重启(原逻辑保留) + // 否则就启动默认参数 return isSettingToEnable ? Service.START_STICKY : super.onStartCommand(intent, flags, startId); } + /** + * 服务连接内部类(Java 7 静态内部类,避免持有外部类强引用导致内存泄漏) + */ private class MyServiceConnection implements ServiceConnection { @Override - public void onServiceConnected(ComponentName name, IBinder service) {} + public void onServiceConnected(ComponentName name, IBinder service) { + // 原逻辑保留(空实现,如需绑定辅助服务可补充具体逻辑) + } @Override public void onServiceDisconnected(ComponentName name) { + // 辅助服务断开时,重新唤醒绑定(原逻辑保留) if (mAppConfigsUtil.isEnableMainService(true)) { wakeupAndBindAssistant(); } } } + /** + * 唤醒并绑定辅助服务(检查服务状态,未存活则启动+绑定) + */ void wakeupAndBindAssistant() { + // 检查辅助服务是否存活(Java 7 静态方法调用,无方法引用) if (!ServiceUtil.isServiceAlive(getApplicationContext(), AssistantService.class.getName())) { + // 启动+绑定辅助服务(Java 7 显式Intent,无Lambda) startService(new Intent(MainService.this, AssistantService.class)); bindService(new Intent(MainService.this, AssistantService.class), mMyServiceConnection, Context.BIND_IMPORTANT); } @@ -663,35 +675,45 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt // ========================================================================= - // GPS相关核心方法 + // GPS相关核心方法(Java 7 语法,匿名内部类实现LocationListener,无Lambda) // ========================================================================= + /** + * 构造函数(Java 7 显式初始化线程池+GPS监听器,无默认构造函数简化) + */ public MainService() { distanceExecutor = Executors.newSingleThreadScheduledExecutor(); initGpsLocationListener(); } + /** + * 初始化GPS监听器(Java 7 匿名内部类实现LocationListener,无Lambda) + */ private void initGpsLocationListener() { LogUtils.d(TAG, "initGpsLocationListener"); mGpsLocationListener = new LocationListener() { @Override public void onLocationChanged(Location location) { if (location != null) { + // 封装GPS位置为PositionModel(Java 7 显式setter调用) PositionModel gpsPos = new PositionModel(); gpsPos.setLatitude(location.getLatitude()); gpsPos.setLongitude(location.getLongitude()); gpsPos.setPositionId("CURRENT_GPS_POS"); gpsPos.setMemo("实时GPS位置"); + // 同步GPS位置+刷新距离+日志(原逻辑保留) syncCurrentGpsPosition(gpsPos); - DistanceCalculatorUtil.getInstance(MainService.this).checkAllTaskTriggerCondition(gpsPos); + DistanceCalculatorUtil.getInstance(MainService.this).checkAllTaskTriggerCondition(gpsPos); LogUtils.d(TAG, "GPS位置更新:纬度=" + location.getLatitude() + ",经度=" + location.getLongitude()); } } @Override public void onStatusChanged(String provider, int status, Bundle extras) { + // 仅处理GPS_PROVIDER状态变化(Java 7 基础判断) if (provider.equals(LocationManager.GPS_PROVIDER)) { String statusDesc = ""; + // 状态枚举判断(Java 7 switch,无增强switch) switch (status) { case LocationProvider.AVAILABLE: statusDesc = "GPS状态:已就绪(可用)"; @@ -711,20 +733,22 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt @Override public void onProviderEnabled(String provider) { + // GPS启用时更新状态+通知+重启定位(Java 7 基础逻辑) if (provider.equals(LocationManager.GPS_PROVIDER)) { isGpsEnabled = true; String statusDesc = "GPS已开启(用户手动打开)"; LogUtils.d(TAG, statusDesc); notifyAllGpsStatusListeners(statusDesc); - updateNotificationGpsStatus("GPS已开启,等待运动状态触发..."); + updateNotificationGpsStatus("GPS已开启,正在获取位置..."); + startGpsLocation(); } } @Override public void onProviderDisabled(String provider) { + // GPS禁用时清空状态+通知+提示(Java 7 基础逻辑) if (provider.equals(LocationManager.GPS_PROVIDER)) { isGpsEnabled = false; - isGpsListening = false; _mCurrentGpsPosition = null; String statusDesc = "GPS已关闭(用户手动关闭)"; LogUtils.w(TAG, statusDesc); @@ -736,15 +760,23 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt }; } + /** + * 检查GPS就绪状态(权限+启用状态,Java 7 基础权限判断,无Stream) + * @return true=GPS就绪,false=未就绪 + */ private boolean checkGpsReady() { + // 检查定位权限(Java 7 基础权限API,无权限请求框架依赖) isGpsPermissionGranted = checkSelfPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) - == PackageManager.PERMISSION_GRANTED; + == PackageManager.PERMISSION_GRANTED; + // 初始化LocationManager(Java 7 显式判断,无Optional) if (mLocationManager == null) { - mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE); + mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE); } + // 检查GPS是否启用(系统LocationManager API,Java 7 兼容) isGpsEnabled = mLocationManager.isProviderEnabled(LocationManager.GPS_PROVIDER); + // 权限未授予:提示+日志+通知 if (!isGpsPermissionGranted) { String tip = "GPS准备失败:缺少精确定位权限"; LogUtils.e(TAG, tip); @@ -753,6 +785,7 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt ToastUtils.show("请授予定位权限,否则无法获取GPS位置"); return false; } + // GPS未启用:提示+日志+通知 if (!isGpsEnabled) { String tip = "GPS准备失败:系统GPS未开启"; LogUtils.e(TAG, tip); @@ -766,24 +799,25 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt return true; } - // 启动GPS持续监听(运动状态回调调用) - public void startGpsLocation() { - if (!checkGpsReady() || isGpsListening) { + /** + * 启动GPS定位(Java 7 异常处理,无try-with-resources,显式捕获SecurityException) + */ + private void startGpsLocation() { + if (!checkGpsReady()) { return; } try { + // 注册GPS位置更新(Java 7 标准LocationManager API,指定Looper为主线程) mLocationManager.requestLocationUpdates( - LocationManager.GPS_PROVIDER, - GPS_UPDATE_INTERVAL, - GPS_UPDATE_DISTANCE, - mGpsLocationListener, - Looper.getMainLooper() + LocationManager.GPS_PROVIDER, + GPS_UPDATE_INTERVAL, + GPS_UPDATE_DISTANCE, + mGpsLocationListener, + Looper.getMainLooper() ); - isGpsListening = true; - - // 获取最后已知位置 + // 获取最后已知GPS位置(缓存位置,避免首次定位等待) Location lastKnownLocation = mLocationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER); if (lastKnownLocation != null) { PositionModel lastGpsPos = new PositionModel(); @@ -800,59 +834,68 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt } } catch (SecurityException e) { + // 定位权限异常(Java 7 显式捕获,无Lambda异常处理) String error = "启动GPS失败(权限异常):" + e.getMessage(); LogUtils.e(TAG, error); notifyAllGpsStatusListeners(error); isGpsPermissionGranted = false; - isGpsListening = false; updateNotificationGpsStatus("定位权限异常,无法获取GPS"); } catch (Exception e) { + // 其他异常(如LocationManager为空、系统服务异常等) String error = "启动GPS失败:" + e.getMessage(); LogUtils.e(TAG, error); notifyAllGpsStatusListeners(error); - isGpsListening = false; updateNotificationGpsStatus("GPS启动失败,尝试重试..."); } } - // 停止GPS持续监听(运动状态回调调用) - public void stopGpsLocation() { + /** + * 停止GPS定位(Java 7 异常处理,移除监听器避免内存泄漏) + */ + private void stopGpsLocation() { + // 校验参数:避免空指针+权限未授予时调用 if (mLocationManager != null && mGpsLocationListener != null && isGpsPermissionGranted) { try { mLocationManager.removeUpdates(mGpsLocationListener); String tip = "GPS定位已停止(移除监听器)"; LogUtils.d(TAG, tip); notifyAllGpsStatusListeners(tip); - isGpsListening = false; } catch (Exception e) { String error = "停止GPS失败:" + e.getMessage(); LogUtils.e(TAG, error); notifyAllGpsStatusListeners(error); - isGpsListening = false; } } } - // 发送任务触发通知 + /** + * 发送任务触发通知(更新前台通知+显示Toast,Java 7 匿名Runnable切换主线程) + * @param task 触发的任务 + * @param bindPos 任务绑定的位置 + * @param currentDistance 当前距离 + */ public void sendTaskTriggerNotification(final PositionTaskModel task, PositionModel bindPos, double currentDistance) { - if (!_mIsServiceRunning) { - return; - } + /*if (!_mIsServiceRunning) { + return; + }*/ + // 格式化通知内容(Java 7 String.format,无TextBlock等Java 15+特性) final String triggerContent = String.format( - "任务触发:%s\n位置:%s\n当前距离:%.1f米(条件:%s%d米)", - task.getTaskDescription(), - bindPos.getMemo(), - currentDistance, - task.isGreaterThan() ? ">" : "<", - task.getDiscussDistance() + "任务触发:%s\n位置:%s\n当前距离:%.1f米(条件:%s%d米)", + task.getTaskDescription(), + bindPos.getMemo(), + currentDistance, + task.isGreaterThan() ? ">" : "<", + task.getDiscussDistance() ); + // 更新前台通知(主线程判断+切换) updateNotificationGpsStatus(triggerContent); + // 显示Toast(主线程安全调用,Java 7 匿名内部类) if (Looper.myLooper() == Looper.getMainLooper()) { ToastUtils.show(triggerContent); - NotificationUtil.show(MainService.this, task.getTaskId(), task.getPositionId(), task.getTaskDescription()); + NotificationUtil.show(MainService.this, task.getTaskId(), task.getPositionId(), task.getTaskDescription()); } else { new Handler(Looper.getMainLooper()).post(new Runnable() { @Override @@ -865,9 +908,14 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt LogUtils.i(TAG, "任务触发通知:" + triggerContent); } - // 更新前台通知的GPS状态 + + /** + * 更新前台通知的GPS状态(Java 7 主线程切换,匿名Runnable实现) + * @param statusText 通知显示的状态文本 + */ void updateNotificationGpsStatus(final String statusText) { if (_mIsServiceRunning) { + // 判断当前线程是否为主线程,避免UI操作在子线程 if (Looper.myLooper() == Looper.getMainLooper()) { NotificationUtil.updateForegroundServiceStatus(this, statusText); } else { @@ -883,8 +931,12 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt // ========================================================================= - // GPS监听通知相关方法 + // GPS监听通知相关方法(Java 7 迭代器遍历弱引用集合,避免内存泄漏) // ========================================================================= + /** + * 通知所有GPS监听者位置更新(Java 7 迭代器+弱引用管理,无Stream) + * @param currentGpsPos 当前最新GPS位置 + */ public void notifyAllGpsListeners(PositionModel currentGpsPos) { if (currentGpsPos == null || mGpsListeners.isEmpty()) { return; @@ -896,89 +948,107 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt GpsUpdateListener listener = ref.get(); if (listener != null) { notifySingleListener(listener, currentGpsPos); - } else { - iter.remove(); - } - } - } - } + } else { + iter.remove(); // 清理已被GC回收的监听者,避免内存泄漏 + } + } + } + } - private void notifySingleListener(final GpsUpdateListener listener, final PositionModel currentGpsPos) { - if (Looper.myLooper() == Looper.getMainLooper()) { - listener.onGpsPositionUpdated(currentGpsPos); - } else { - new Handler(Looper.getMainLooper()).post(new Runnable() { + /** + * 通知单个GPS监听者位置更新(主线程安全,Java 7 匿名Runnable) + * @param listener 单个监听者 + * @param currentGpsPos 当前GPS位置 + */ + private void notifySingleListener(final GpsUpdateListener listener, final PositionModel currentGpsPos) { + if (Looper.myLooper() == Looper.getMainLooper()) { + listener.onGpsPositionUpdated(currentGpsPos); + } else { + new Handler(Looper.getMainLooper()).post(new Runnable() { @Override public void run() { listener.onGpsPositionUpdated(currentGpsPos); } }); - } - } + } + } - private void notifyAllGpsStatusListeners(final String status) { - if (status == null || mGpsListeners.isEmpty()) { - return; - } - synchronized (mListenerLock) { - Iterator> iter = mGpsListeners.iterator(); - while (iter.hasNext()) { - WeakReference ref = iter.next(); - final GpsUpdateListener listener = ref.get(); - if (listener != null) { - if (Looper.myLooper() == Looper.getMainLooper()) { - listener.onGpsStatusChanged(status); - } else { - new Handler(Looper.getMainLooper()).post(new Runnable() { + /** + * 通知所有GPS监听者状态变化(如GPS开启/关闭、信号弱等,Java 7 迭代器) + * @param status GPS状态描述文本 + */ + private void notifyAllGpsStatusListeners(final String status) { + if (status == null || mGpsListeners.isEmpty()) { + return; + } + synchronized (mListenerLock) { + Iterator> iter = mGpsListeners.iterator(); + while (iter.hasNext()) { + WeakReference ref = iter.next(); + final GpsUpdateListener listener = ref.get(); + if (listener != null) { + // 主线程切换,避免监听者在子线程处理UI + if (Looper.myLooper() == Looper.getMainLooper()) { + listener.onGpsStatusChanged(status); + } else { + new Handler(Looper.getMainLooper()).post(new Runnable() { @Override public void run() { listener.onGpsStatusChanged(status); } }); - } - } else { - iter.remove(); - } - } - } - } + } + } else { + iter.remove(); // 清理无效弱引用 + } + } + } + } - public void registerGpsUpdateListener(GpsUpdateListener listener) { - if (listener == null) { - LogUtils.w(TAG, "registerGpsUpdateListener:监听者为空"); - return; - } - synchronized (mListenerLock) { - mGpsListeners.add(new WeakReference(listener)); - LogUtils.d(TAG, "GPS监听注册成功,当前数量:" + mGpsListeners.size()); - if (_mCurrentGpsPosition != null) { - notifySingleListener(listener, _mCurrentGpsPosition); - } - } - } + /** + * 注册GPS更新监听(Java 7 弱引用添加,避免监听者内存泄漏) + * @param listener GPS更新监听者(如Activity/Adapter) + */ + public void registerGpsUpdateListener(GpsUpdateListener listener) { + if (listener == null) { + LogUtils.w(TAG, "registerGpsUpdateListener:监听者为空"); + return; + } + synchronized (mListenerLock) { + mGpsListeners.add(new WeakReference(listener)); + LogUtils.d(TAG, "GPS监听注册成功,当前数量:" + mGpsListeners.size()); + // 注册后立即推送当前GPS位置(避免监听者错过初始数据) + if (_mCurrentGpsPosition != null) { + notifySingleListener(listener, _mCurrentGpsPosition); + } + } + } - public void unregisterGpsUpdateListener(GpsUpdateListener listener) { - if (listener == null) { - LogUtils.w(TAG, "unregisterGpsUpdateListener:监听者为空"); - return; - } - synchronized (mListenerLock) { - Iterator> iter = mGpsListeners.iterator(); - while (iter.hasNext()) { - WeakReference ref = iter.next(); - if (ref.get() == listener || ref.get() == null) { - iter.remove(); - LogUtils.d(TAG, "GPS监听反注册成功,当前数量:" + mGpsListeners.size()); - break; - } - } - } - } + /** + * 反注册GPS更新监听(Java 7 迭代器清理,避免内存泄漏) + * @param listener 待反注册的GPS监听者 + */ + public void unregisterGpsUpdateListener(GpsUpdateListener listener) { + if (listener == null) { + LogUtils.w(TAG, "unregisterGpsUpdateListener:监听者为空"); + return; + } + synchronized (mListenerLock) { + Iterator> iter = mGpsListeners.iterator(); + while (iter.hasNext()) { + WeakReference ref = iter.next(); + // 匹配目标监听者或已回收的弱引用,直接移除 + if (ref.get() == listener || ref.get() == null) { + iter.remove(); + LogUtils.d(TAG, "GPS监听反注册成功,当前数量:" + mGpsListeners.size()); + break; + } + } + } + } - -// ========================================================================= -// LocalBinder 定义(无修改) -// ========================================================================= + // 补全 LocalBinder 定义(与 LocationActivity 中的 LocalBinder 保持一致) + // 注意:若 LocationActivity 已定义 LocalBinder,此处可删除;建议统一在 MainService 中定义,避免重复 public class LocalBinder extends android.os.Binder { private MainService mService; @@ -990,7 +1060,7 @@ public class MainService extends Service implements LocalMotionDetector.MotionSt return mService; } } + } - diff --git a/positions/src/main/java/cc/winboll/studio/positions/utils/APPPlusUtils.java b/positions/src/main/java/cc/winboll/studio/positions/utils/APPPlusUtils.java deleted file mode 100644 index d747445..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/utils/APPPlusUtils.java +++ /dev/null @@ -1,194 +0,0 @@ -package cc.winboll.studio.positions.utils; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/10 09:51 - * @Describe 应用图标切换工具类(启用组件时创建对应快捷方式) - */ -import android.content.ComponentName; -import android.content.Context; -import android.content.Intent; -import android.content.pm.PackageManager; -import android.os.Build; -import android.widget.Toast; -import cc.winboll.studio.libappbase.LogUtils; -import cc.winboll.studio.positions.App; -import cc.winboll.studio.positions.MainActivity; - -public class APPPlusUtils { - public static final String TAG = "APPPlusUtils"; - - // 快捷方式配置(名称+图标,需与实际资源匹配) -// private static final String PLUS_SHORTCUT_NAME = "位置服务-Laojun"; -// private static final int PLUS_SHORTCUT_ICON = R.mipmap.ic_launcher; // Laojun 图标资源 - - /** - * 添加Plus组件与图标 - */ - public static boolean openAPPPlus(Context context) { - if (context == null) { - LogUtils.d(TAG, "切换失败:上下文为空"); - Toast.makeText(context, "图标切换失败", Toast.LENGTH_SHORT).show(); - return false; - } - - PackageManager pm = context.getPackageManager(); - ComponentName plusComponentLaojun = new ComponentName(context, App.COMPONENT_LAOJUN); - //ComponentName plusComponentWuKong = new ComponentName(context, MainActivity.COMPONENT_WUKONG); - - try { - //disableComponent(pm, plusComponentWuKong); - enableComponent(pm, plusComponentLaojun); - - // 2. 创建 Laojun 组件对应的快捷方式(自动去重) -// boolean shortcutCreated = createComponentShortcut(context, plusComponent, PLUS_SHORTCUT_NAME, PLUS_SHORTCUT_ICON); -// -// // 3. 通知桌面刷新图标 -// context.sendBroadcast(new Intent(Intent.ACTION_PACKAGE_CHANGED) -// .setData(android.net.Uri.parse("package:" + context.getPackageName()))); -// -// // 4. 反馈结果 -// String logMsg = shortcutCreated ? "启用 Laojun + 快捷方式创建成功" : "启用 Laojun 成功,快捷方式创建失败"; -// String toastMsg = shortcutCreated ? "图标切换为 Laojun,已创建快捷方式" : "图标切换为 Laojun,快捷方式创建失败"; -// LogUtils.d(TAG, logMsg); -// Toast.makeText(context, toastMsg, Toast.LENGTH_SHORT).show(); -// - return true; - - } catch (Exception e) { - LogUtils.e(TAG, "Laojun 图标切换失败:" + e.getMessage()); - // 失败兜底:启用 Wukong 组件 - //enableComponent(pm, wukongComponent); - Toast.makeText(context, "图标切换失败" + e.getMessage(), Toast.LENGTH_SHORT).show(); - return false; - } - } - - - /** - * 移除Plus组件 - */ - public static boolean closeAPPPlus(Context context) { - if (context == null) { - LogUtils.d(TAG, "切换失败:上下文为空"); - Toast.makeText(context, "图标切换失败", Toast.LENGTH_SHORT).show(); - return false; - } - - PackageManager pm = context.getPackageManager(); - ComponentName plusComponentLaojun = new ComponentName(context, App.COMPONENT_LAOJUN); - //ComponentName plusComponentWuKong = new ComponentName(context, MainActivity.COMPONENT_WUKONG); - - disableComponent(pm, plusComponentLaojun); - //enableComponent(pm, plusComponentWuKong); - - return true; - } - - /** - * 创建指定组件的桌面快捷方式(自动去重,兼容 Android 8.0+) - * @param component 目标组件(如 LAOJUN_ACTIVITY) - * @param name 快捷方式名称 - * @param iconRes 快捷方式图标资源ID - * @return 是否创建成功 - */ - private static boolean createComponentShortcut(Context context, ComponentName component, String name, int iconRes) { - if (context == null || component == null || name == null || iconRes == 0) { - LogUtils.d(TAG, "快捷方式创建失败:参数为空"); - return false; - } - - // Android 8.0+(API 26+):使用 ShortcutManager(系统推荐) - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { - try { - PackageManager pm = context.getPackageManager(); - android.content.pm.ShortcutManager shortcutManager = context.getSystemService(android.content.pm.ShortcutManager.class); - if (shortcutManager == null || !shortcutManager.isRequestPinShortcutSupported()) { - LogUtils.d(TAG, "系统不支持创建快捷方式"); - return false; - } - - // 检查是否已存在该组件的快捷方式(去重) - for (android.content.pm.ShortcutInfo info : shortcutManager.getPinnedShortcuts()) { - if (component.getClassName().equals(info.getIntent().getComponent().getClassName())) { - LogUtils.d(TAG, "快捷方式已存在:" + component.getClassName()); - return true; - } - } - - // 构建启动目标组件的意图 - Intent launchIntent = new Intent(Intent.ACTION_MAIN) - .setComponent(component) - .addCategory(Intent.CATEGORY_LAUNCHER) - .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP); - - // 构建快捷方式信息 - android.content.pm.ShortcutInfo shortcutInfo = new android.content.pm.ShortcutInfo.Builder(context, component.getClassName()) - .setShortLabel(name) - .setLongLabel(name) - .setIcon(android.graphics.drawable.Icon.createWithResource(context, iconRes)) - .setIntent(launchIntent) - .build(); - - // 请求创建快捷方式(需用户确认) - shortcutManager.requestPinShortcut(shortcutInfo, null); - return true; - - } catch (Exception e) { - LogUtils.d(TAG, "Android O+ 快捷方式创建失败:" + e.getMessage()); - return false; - } - } else { - // Android 8.0 以下:使用广播(兼容旧机型) - try { - // 构建启动目标组件的意图 - Intent launchIntent = new Intent(Intent.ACTION_MAIN) - .setComponent(component) - .addCategory(Intent.CATEGORY_LAUNCHER) - .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP); - - // 构建创建快捷方式的广播意图 - Intent installIntent = new Intent("com.android.launcher.action.INSTALL_SHORTCUT"); - installIntent.putExtra(Intent.EXTRA_SHORTCUT_INTENT, launchIntent); - installIntent.putExtra(Intent.EXTRA_SHORTCUT_NAME, name); - installIntent.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE, - Intent.ShortcutIconResource.fromContext(context, iconRes)); - installIntent.putExtra("duplicate", false); // 禁止重复创建 - - context.sendBroadcast(installIntent); - return true; - - } catch (Exception e) { - LogUtils.d(TAG, "Android O- 快捷方式创建失败:" + e.getMessage()); - return false; - } - } - } - - /** - * 启用组件(带状态检查,避免重复操作) - */ - private static void enableComponent(PackageManager pm, ComponentName component) { - if (pm.getComponentEnabledSetting(component) != PackageManager.COMPONENT_ENABLED_STATE_ENABLED) { - pm.setComponentEnabledSetting( - component, - PackageManager.COMPONENT_ENABLED_STATE_ENABLED, - PackageManager.DONT_KILL_APP | PackageManager.SYNCHRONOUS - ); - } - } - - /** - * 禁用组件(带状态检查,避免重复操作) - */ - private static void disableComponent(PackageManager pm, ComponentName component) { - if (pm.getComponentEnabledSetting(component) != PackageManager.COMPONENT_ENABLED_STATE_DISABLED) { - pm.setComponentEnabledSetting( - component, - PackageManager.COMPONENT_ENABLED_STATE_DISABLED, - PackageManager.DONT_KILL_APP | PackageManager.SYNCHRONOUS - ); - } - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/utils/ActivityAliasUtils.java b/positions/src/main/java/cc/winboll/studio/positions/utils/ActivityAliasUtils.java deleted file mode 100644 index 619e5e6..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/utils/ActivityAliasUtils.java +++ /dev/null @@ -1,148 +0,0 @@ -package cc.winboll.studio.positions.utils; - -import android.content.Context; -import android.content.Intent; -import android.content.pm.ActivityInfo; -import android.content.pm.PackageManager; -import android.text.TextUtils; -import cc.winboll.studio.libappbase.LogUtils; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/15 15:23 - * @Describe Activity Alias 工具类(兼容 Android 所有版本,Java 7 语法) - * 用于获取 activity-alias 对应的原始 Activity 组件名、判断 alias 类型、获取启动组件名等 - */ -public class ActivityAliasUtils { - private static final String TAG = "ActivityAliasUtils"; - - /** - * 获取 activity-alias 指向的原始 Activity 组件名 - * - * @param context 上下文(建议用 ApplicationContext) - * @param aliasName activity-alias 的组件名(完整路径,如 ".AliasActivity" 或 "com.winboll.app.AliasActivity") - * @return 原始 Activity 的完整组件名(如 "com.winboll.app.OriginalActivity"),失败返回 null - */ - public static String getTargetActivityName(Context context, String aliasName) { - // 校验参数 - if (context == null || TextUtils.isEmpty(aliasName)) { - LogUtils.e(TAG, "getTargetActivityName: context is null or aliasName is empty"); - return null; - } - - // 补全组件名(若传入的是短名,自动拼接包名) - String fullAliasName = aliasName.startsWith(".") - ? context.getPackageName() + aliasName - : aliasName; - - try { - // 1. 获取 PackageManager - PackageManager packageManager = context.getPackageManager(); - - // 2. 解析 activity-alias 的 ActivityInfo(flag 必须设为 PackageManager.GET_META_DATA,否则可能获取不到 targetActivity) - ActivityInfo aliasActivityInfo = packageManager.getActivityInfo( - new android.content.ComponentName(context.getPackageName(), fullAliasName), - PackageManager.GET_META_DATA - ); - - // 3. 获取 targetActivity(原始 Activity 组件名) - String targetActivity = aliasActivityInfo.targetActivity; - if (TextUtils.isEmpty(targetActivity)) { - LogUtils.e(TAG, "getTargetActivityName: targetActivity is empty for alias " + fullAliasName); - return null; - } - - // 4. 补全原始 Activity 的完整包名(若 targetActivity 是短名) - String fullTargetName = targetActivity.startsWith(".") - ? context.getPackageName() + targetActivity - : targetActivity; - - LogUtils.d(TAG, "getTargetActivityName: alias=" + fullAliasName + ", target=" + fullTargetName); - return fullTargetName; - - } catch (PackageManager.NameNotFoundException e) { - LogUtils.e(TAG, "getTargetActivityName: alias not found - " + fullAliasName, e); - } catch (Exception e) { - LogUtils.e(TAG, "getTargetActivityName: unknown error", e); - } - return null; - } - - /** - * 判断某个组件名是否为 activity-alias(而非原始 Activity) - * - * @param context 上下文 - * @param componentName 待判断的组件名(完整路径) - * @return true:是 activity-alias;false:不是或判断失败 - */ - public static boolean isActivityAlias(Context context, String componentName) { - // 调用 getTargetActivityName,若返回非空,则说明是 alias - return !TextUtils.isEmpty(getTargetActivityName(context, componentName)); - } - - /** - * 从启动的 Intent 中获取实际的目标组件名(处理 alias 场景) - * 适用于 Activity 中获取自身真实组件名(原始 Activity 名) - * - * @param context 当前 Activity 上下文 - * @return 真实的目标组件名(原始 Activity 名,若为 alias 启动则返回原始 Activity,否则返回自身) - */ - public static String getRealTargetNameFromIntent(Context context) { - if (context == null) { - LogUtils.e(TAG, "getRealTargetNameFromIntent: context is null"); - return null; - } - - // 获取当前 Activity 的组件名(可能是 alias) - String currentComponentName = context.getClass().getName(); - // 检查是否为 alias,若是则返回 target,否则返回自身 - String targetName = getTargetActivityName(context, currentComponentName); - return TextUtils.isEmpty(targetName) ? currentComponentName : targetName; - } - - /** - * 获取当前活动上下文(Activity)的启动组件名(即启动时使用的组件名,可能是 alias 或原始 Activity) - * 场景:若通过 alias 启动 Activity,返回 alias 名;若直接启动原始 Activity,返回原始 Activity 名 - * - * @param context 当前 Activity 上下文(必须是 Activity 实例,不能是 ApplicationContext) - * @return 启动组件的完整名,失败返回 null - */ - public static String getLaunchComponentName(Context context) { - // 1. 校验上下文类型(必须是 Activity,否则无法获取启动 Intent) - if (context == null) { - LogUtils.e(TAG, "getLaunchComponentName: context is null"); - return null; - } - if (!(context instanceof android.app.Activity)) { - LogUtils.e(TAG, "getLaunchComponentName: context must be Activity instance, current is " + context.getClass().getName()); - return null; - } - - try { - // 2. 获取启动当前 Activity 的 Intent - android.app.Activity activity = (android.app.Activity) context; - Intent launchIntent = activity.getIntent(); - if (launchIntent == null) { - LogUtils.e(TAG, "getLaunchComponentName: launch Intent is null"); - return null; - } - - // 3. 从 Intent 中获取启动组件名(ComponentName) - android.content.ComponentName componentName = launchIntent.getComponent(); - if (componentName == null) { - LogUtils.e(TAG, "getLaunchComponentName: ComponentName is null in launch Intent"); - return null; - } - - // 4. 获取组件的完整类名(即启动时使用的组件名) - String launchComponentName = componentName.getClassName(); - LogUtils.d(TAG, "getLaunchComponentName: current launch component is " + launchComponentName); - return launchComponentName; - - } catch (Exception e) { - LogUtils.e(TAG, "getLaunchComponentName: failed to get launch component name", e); - return null; - } - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/utils/AppConfigsUtil.java b/positions/src/main/java/cc/winboll/studio/positions/utils/AppConfigsUtil.java index 97b728f..3110de7 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/utils/AppConfigsUtil.java +++ b/positions/src/main/java/cc/winboll/studio/positions/utils/AppConfigsUtil.java @@ -1,8 +1,6 @@ package cc.winboll.studio.positions.utils; import android.content.Context; -import cc.winboll.studio.positions.AppLevel; import cc.winboll.studio.positions.models.AppConfigsModel; -import cc.winboll.studio.positions.App; /** * @Author ZhanGSKen&豆包大模型 @@ -60,27 +58,11 @@ public class AppConfigsUtil { } public void setIsEnableMainService(boolean isEnableMainService) { - if (mAppConfigsModel == null) { + if(mAppConfigsModel == null) { mAppConfigsModel = new AppConfigsModel(); } mAppConfigsModel.setIsEnableMainService(isEnableMainService); saveConfigs(); } - - public AppLevel getAppLevel(boolean isReloadConfigs) { - if (isReloadConfigs) { - loadConfigs(); - } - return (mAppConfigsModel == null) ?AppLevel.WUKONG: mAppConfigsModel.getAppLevel(); - } - - public void setAppLevel(AppLevel appLevel) { - if (mAppConfigsModel == null) { - mAppConfigsModel = new AppConfigsModel(); - } - App._mAppLevel = appLevel; - mAppConfigsModel.setAppLevel(appLevel); - saveConfigs(); - } } diff --git a/positions/src/main/java/cc/winboll/studio/positions/utils/JsonShareHandler.java b/positions/src/main/java/cc/winboll/studio/positions/utils/JsonShareHandler.java deleted file mode 100644 index 13095c6..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/utils/JsonShareHandler.java +++ /dev/null @@ -1,241 +0,0 @@ -package cc.winboll.studio.positions.utils; - -import android.app.AlertDialog; -import android.content.Context; -import android.content.DialogInterface; -import android.content.Intent; -import android.database.Cursor; -import android.net.Uri; -import android.os.Build; -import android.os.Environment; -import android.provider.MediaStore; -import android.util.Log; -import android.widget.Toast; -import java.io.File; -import java.io.FileOutputStream; -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/13 15:42 - * @Describe JsonShareHandler - * 外部 JSON 文件分享处理工具类 - * 功能:接收外部分享的 .json 文件,弹出确认对话框,保存到外部存储 files/BaseBean 目录 - */ -public class JsonShareHandler { - private static final String TAG = "JsonShareHandler"; - private static final String TARGET_DIR = "BaseBean"; - private static final String MIME_TYPE_JSON = "application/json"; - private static final String FILE_SUFFIX_JSON = ".json"; - // 对话框回调接口(Java7 无 Lambda,用接口实现) - public interface ConfirmCallback { - void onConfirm(boolean isConfirm); - } - - /** - * 处理外部分享的 Intent,先弹出确认对话框,再决定是否接收文件 - * @param context 上下文(需为 Activity,否则无法弹出对话框) - * @param intent 分享 Intent - * @param callback 确认结果回调(用于 Activity 处理后续逻辑) - */ - public static void handleSharedJsonWithConfirm(final Context context, final Intent intent, final ConfirmCallback callback) { - if (context == null || intent == null || callback == null) { - Log.e(TAG, "参数为空,处理失败"); - if (callback != null) callback.onConfirm(false); - return; - } - - // 1. 先验证 Intent 合法性(提前过滤无效分享) - String action = intent.getAction(); - String type = intent.getType(); - if (!Intent.ACTION_SEND.equals(action) || type == null) { - Log.e(TAG, "非文件分享 Intent"); - Toast.makeText(context, "不支持的分享类型", Toast.LENGTH_SHORT).show(); - callback.onConfirm(false); - return; - } - - // 2. 弹出确认对话框 - new AlertDialog.Builder(context) - .setTitle("接收 JSON 文件") - .setMessage("是否接收并保存该 JSON 文件?") - .setPositiveButton("Yes", new DialogInterface.OnClickListener() { - @Override - public void onClick(DialogInterface dialog, int which) { - dialog.dismiss(); - // 3. 点击 Yes,处理文件保存 - String savedPath = handleSharedJsonFile(context, intent); - if (savedPath != null) { - Toast.makeText(context, "文件保存成功:" + savedPath, Toast.LENGTH_LONG).show(); - callback.onConfirm(true); - } else { - Toast.makeText(context, "文件保存失败", Toast.LENGTH_SHORT).show(); - callback.onConfirm(false); - } - } - }) - .setNegativeButton("No", new DialogInterface.OnClickListener() { - @Override - public void onClick(DialogInterface dialog, int which) { - dialog.dismiss(); - // 4. 点击 No,直接退出处理 - callback.onConfirm(false); - } - }) - .setCancelable(false) // 不可点击外部取消 - .show(); - } - - /** - * 核心文件处理逻辑(原有功能,无修改) - */ - private static String handleSharedJsonFile(Context context, Intent intent) { - String action = intent.getAction(); - String type = intent.getType(); - - // 验证 JSON 格式 - if (!MIME_TYPE_JSON.equals(type) && !type.contains("json")) { - Uri uri = intent.getParcelableExtra(Intent.EXTRA_STREAM); - if (uri == null || !getFileNameFromUri(context, uri).endsWith(FILE_SUFFIX_JSON)) { - Log.e(TAG, "接收的文件不是 JSON 格式"); - return null; - } - } - - Uri sharedUri = intent.getParcelableExtra(Intent.EXTRA_STREAM); - if (sharedUri == null) { - Log.e(TAG, "未获取到分享的文件 Uri"); - return null; - } - - try { - // 创建保存目录 - File saveDir = getTargetSaveDir(context); - if (!saveDir.exists() && !saveDir.mkdirs()) { - Log.e(TAG, "创建保存目录失败:" + saveDir.getAbsolutePath()); - return null; - } - - // 解析文件名(兼容低版本) - String fileName = getFileNameFromUri(context, sharedUri); - if (fileName == null || !fileName.endsWith(FILE_SUFFIX_JSON)) { - fileName = "default_" + System.currentTimeMillis() + FILE_SUFFIX_JSON; - Log.w(TAG, "文件名解析失败,使用默认名称:" + fileName); - } - - // 复制文件 - File targetFile = new File(saveDir, fileName); - boolean copySuccess = copyFileFromUri(context, sharedUri, targetFile); - return copySuccess ? targetFile.getAbsolutePath() : null; - - } catch (Exception e) { - Log.e(TAG, "处理分享文件异常:" + e.getMessage()); - return null; - } - } - - /** - * 获取目标保存目录(兼容 Android 10+ 分区存储) - */ - private static File getTargetSaveDir(Context context) { - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { - return new File(context.getExternalFilesDir(null), TARGET_DIR); - } else { - return new File( - Environment.getExternalStorageDirectory() + File.separator + - "Android" + File.separator + - "data" + File.separator + - context.getPackageName() + File.separator + - "files" + File.separator + - TARGET_DIR - ); - } - } - - /** - * 从 Uri 解析文件名(兼容所有 Android 版本) - */ - private static String getFileNameFromUri(Context context, Uri uri) { - if (uri == null) return null; - - // 1. 文件 Uri(file:// 开头) - if ("file".equals(uri.getScheme())) { - return new File(uri.getPath()).getName(); - } - - // 2. 内容 Uri(content:// 开头) - if ("content".equals(uri.getScheme())) { - Cursor cursor = null; - try { - cursor = context.getContentResolver().query( - uri, - new String[]{MediaStore.MediaColumns.DISPLAY_NAME}, - null, - null, - null - ); - if (cursor != null && cursor.moveToFirst()) { - int nameIndex = cursor.getColumnIndex(MediaStore.MediaColumns.DISPLAY_NAME); - if (nameIndex != -1) { - return cursor.getString(nameIndex); - } - } - } catch (Exception e) { - Log.e(TAG, "解析内容 Uri 文件名失败:" + e.getMessage()); - } finally { - if (cursor != null) cursor.close(); - } - } - - // 3. 解析失败,返回默认名称 - String lastPathSegment = uri.getLastPathSegment(); - return lastPathSegment != null ? lastPathSegment : "unknown.json"; - } - - /** - * 复制 Uri 指向的文件到目标路径 - */ - private static boolean copyFileFromUri(Context context, Uri sourceUri, File targetFile) { - InputStream inputStream = null; - OutputStream outputStream = null; - try { - inputStream = context.getContentResolver().openInputStream(sourceUri); - if (inputStream == null) { - Log.e(TAG, "无法打开源文件输入流"); - return false; - } - - outputStream = new FileOutputStream(targetFile); - byte[] buffer = new byte[1024 * 4]; - int length; - while ((length = inputStream.read(buffer)) != -1) { - outputStream.write(buffer, 0, length); - } - outputStream.flush(); - Log.d(TAG, "文件保存成功:" + targetFile.getAbsolutePath()); - return true; - - } catch (IOException e) { - Log.e(TAG, "文件复制异常:" + e.getMessage()); - return false; - } finally { - try { - if (inputStream != null) inputStream.close(); - if (outputStream != null) outputStream.close(); - } catch (IOException e) { - Log.e(TAG, "关闭流异常:" + e.getMessage()); - } - } - } - - /** - * 检查外部存储是否可用 - */ - public static boolean isExternalStorageAvailable() { - String state = Environment.getExternalStorageState(); - return Environment.MEDIA_MOUNTED.equals(state); - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/utils/LocalMotionDetector.java b/positions/src/main/java/cc/winboll/studio/positions/utils/LocalMotionDetector.java deleted file mode 100644 index 0e2a186..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/utils/LocalMotionDetector.java +++ /dev/null @@ -1,168 +0,0 @@ -package cc.winboll.studio.positions.utils; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/05 15:49 - * @Describe LocalMotionDetector - */ -import android.content.Context; -import android.hardware.Sensor; -import android.hardware.SensorEvent; -import android.hardware.SensorEventListener; -import android.hardware.SensorManager; -import android.os.Handler; -import android.os.Looper; -import android.util.Log; -import cc.winboll.studio.libappbase.LogUtils; - -/** - * 本机运动状态监测工具(无联网,纯传感器) - */ -public class LocalMotionDetector implements SensorEventListener { - public static final String TAG = "LocalMotionDetector"; - - // 配置参数(重点修改:调高运动阈值,适配坐立持机场景) - private static final float MOTION_THRESHOLD = 1.8f; // 从0.5f调高到1.8f(过滤坐立轻微晃动) - private static final long STATUS_CHECK_INTERVAL = 3000; // 3秒判断一次状态 - private static final int STEP_CHANGE_THRESHOLD = 2; // 3秒≥2步判定行走 - - private SensorManager mSensorManager; - private Sensor mAccelerometer; - private Sensor mStepCounter; - private Handler mMainHandler; - private MotionStatusCallback mCallback; - - private boolean mIsDetecting = false; - private float mLastAccelMagnitude = 0f; - private int mLastStepCount = 0; - private int mCurrentStepCount = 0; - private boolean mIsWalking = false; - - // 单例模式 - private static LocalMotionDetector sInstance; - public static LocalMotionDetector getInstance() { - if (sInstance == null) { - synchronized (LocalMotionDetector.class) { - if (sInstance == null) { - sInstance = new LocalMotionDetector(); - } - } - } - return sInstance; - } - - private LocalMotionDetector() { - mMainHandler = new Handler(Looper.getMainLooper()); - } - - /** - * 开始监测运动状态 - */ - public void startDetection(Context context, MotionStatusCallback callback) { - if (mIsDetecting) return; - mCallback = callback; - mSensorManager = (SensorManager) context.getSystemService(Context.SENSOR_SERVICE); - - // 初始化传感器 - mAccelerometer = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER); - mStepCounter = mSensorManager.getDefaultSensor(Sensor.TYPE_STEP_COUNTER); - - // 注册传感器监听 - if (mAccelerometer != null) { - mSensorManager.registerListener(this, mAccelerometer, SensorManager.SENSOR_DELAY_NORMAL, mMainHandler); - } - if (mStepCounter != null) { - mSensorManager.registerListener(this, mStepCounter, SensorManager.SENSOR_DELAY_NORMAL, mMainHandler); - LogUtils.d(TAG, "计步传感器已启动"); - } else { - LogUtils.d(TAG, "设备不支持计步传感器,仅用加速度判断"); - } - - // 启动定时状态检测 - mMainHandler.postDelayed(mStatusCheckRunnable, STATUS_CHECK_INTERVAL); - mIsDetecting = true; - LogUtils.d(TAG, "运动状态监测已启动"); - } - - /** - * 停止监测 - */ - public void stopDetection() { - if (!mIsDetecting) return; - if (mSensorManager != null) { - mSensorManager.unregisterListener(this); - } - mMainHandler.removeCallbacksAndMessages(null); - mIsDetecting = false; - mIsWalking = false; - mCallback = null; - LogUtils.d(TAG, "运动状态监测已停止"); - } - - @Override - public void onSensorChanged(SensorEvent event) { - if (!mIsDetecting) return; - switch (event.sensor.getType()) { - case Sensor.TYPE_ACCELEROMETER: - // 计算加速度幅度(保留原逻辑,阈值已调高) - float accelX = Math.abs(event.values[0]); - float accelY = Math.abs(event.values[1]); - float accelZ = Math.abs(event.values[2]); - mLastAccelMagnitude = accelX + accelY + accelZ; - break; - case Sensor.TYPE_STEP_COUNTER: - // 累计步数 - mCurrentStepCount = (int) event.values[0]; - break; - } - } - - /** - * 定时判断运动状态(优化逻辑:计步为0时,即使有轻微加速度也判定为静止) - */ - private final Runnable mStatusCheckRunnable = new Runnable() { - @Override - public void run() { - if (!mIsDetecting || mCallback == null) return; - - //LogUtils.d(TAG, "mStatusCheckRunnable run"); - - boolean newIsWalking = false; - // 结合计步器+加速度判断(优化:优先计步,无步数时严格按高阈值判断) - if (mStepCounter != null) { - int stepChange = mCurrentStepCount - mLastStepCount; - // 只有“步数达标” 或 “无步数但加速度远超坐立幅度”,才判定为行走 - newIsWalking = (stepChange >= STEP_CHANGE_THRESHOLD) - && (mLastAccelMagnitude >= MOTION_THRESHOLD); // 增加步数+加速度双重校验 - mLastStepCount = mCurrentStepCount; - } else { - // 无计步器时,仅用高阈值判断 - newIsWalking = mLastAccelMagnitude >= MOTION_THRESHOLD; - } - - // 状态变化时回调 - if (newIsWalking != mIsWalking) { - mIsWalking = newIsWalking; - String statusDesc = mIsWalking ? "行走状态" : "静止/低运动状态"; - LogUtils.d(TAG, "运动状态变化:" + statusDesc + " | 加速度幅度:" + mLastAccelMagnitude); // 增加日志便于调试 - mCallback.onMotionStatusChanged(mIsWalking, statusDesc); - } - - LogUtils.d(TAG, String.format("运动状态 newIsWalking %s", newIsWalking)); - - // 循环检测 - mMainHandler.postDelayed(this, STATUS_CHECK_INTERVAL); - } - }; - - @Override - public void onAccuracyChanged(Sensor sensor, int accuracy) {} - - /** - * 运动状态回调接口 - */ - public interface MotionStatusCallback { - void onMotionStatusChanged(boolean isWalking, String statusDesc); - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/utils/MyActivityLifecycleCallbacks.java b/positions/src/main/java/cc/winboll/studio/positions/utils/MyActivityLifecycleCallbacks.java deleted file mode 100644 index cfeb878..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/utils/MyActivityLifecycleCallbacks.java +++ /dev/null @@ -1,105 +0,0 @@ -package cc.winboll.studio.positions.utils; - -import android.app.Activity; -import android.app.Application; -import android.content.Intent; -import android.os.Bundle; -import cc.winboll.studio.libappbase.LogUtils; -import cc.winboll.studio.libappbase.ToastUtils; -import cc.winboll.studio.positions.PointLevel; -import cc.winboll.studio.positions.activities.WinBoLLActivity; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/15 15:59 - * @Describe 应用活动窗口状态响应类 - * 主要用于设置应用级别与组件状态 - */ -public class MyActivityLifecycleCallbacks implements Application.ActivityLifecycleCallbacks { - - public static final String TAG = "MyActivityLifecycleCallbacks"; - - public String mInfo = ""; - - public MyActivityLifecycleCallbacks() { - - } - - void createActivityeInfo(Activity activity) { - StringBuilder sb = new StringBuilder(); - Intent receivedIntent = activity.getIntent(); - sb.append("\nCallingActivity : \n"); - if (activity.getCallingActivity() != null) { - sb.append(activity.getCallingActivity().getPackageName()); - } - sb.append("\nReceived Intent Package : \n"); - sb.append(receivedIntent.getPackage()); - - Bundle extras = receivedIntent.getExtras(); - if (extras != null) { - for (String key : extras.keySet()) { - sb.append("\nIntentInfo"); - sb.append("\n键: "); - sb.append(key); - sb.append(", 值: "); - sb.append(extras.get(key)); - //Log.d("IntentInfo", "键: " + key + ", 值: " + extras.get(key)); - } - } - mInfo = sb.toString(); - //Log.d("IntentInfo", "发送Intent的应用包名: " + senderPackage); - } - - public void showActivityeInfo() { - //ToastUtils.show("ActivityeInfo : " + mInfo); - LogUtils.d(TAG, "ActivityeInfo : " + mInfo); - } - - @Override - public void onActivityCreated(Activity activity, Bundle savedInstanceState) { - // 在这里可以做一些初始化相关的操作,例如记录Activity的创建时间等 - //System.out.println(activity.getLocalClassName() + " was created"); - LogUtils.d(TAG, activity.getLocalClassName() + " was created"); - createActivityeInfo(activity); - } - - @Override - public void onActivityStarted(Activity activity) { - //System.out.println(activity.getLocalClassName() + " was started"); - LogUtils.d(TAG, activity.getLocalClassName() + " was started"); - //createActivityeInfo(activity); - } - - @Override - public void onActivityResumed(Activity activity) { - //System.out.println(activity.getLocalClassName() + " was resumed"); - LogUtils.d(TAG, activity.getLocalClassName() + " was resumed"); - //createActivityeInfo(activity); - } - - @Override - public void onActivityPaused(Activity activity) { - ToastUtils.show("Activity Paused"); - // 应用从正在活动状态抽离出来时,设置应用入口级别状态,设置为时空虚幻而不确定的哆啦A梦级别。 - WinBoLLActivity._mPointLevel = PointLevel.DORAEMON; - //System.out.println(activity.getLocalClassName() + " was paused"); - LogUtils.d(TAG, activity.getLocalClassName() + " was paused"); - } - - @Override - public void onActivityStopped(Activity activity) { - //System.out.println(activity.getLocalClassName() + " was stopped"); - LogUtils.d(TAG, activity.getLocalClassName() + " was stopped"); - } - - @Override - public void onActivitySaveInstanceState(Activity activity, Bundle outState) { - // 可以在这里添加保存状态的自定义逻辑 - } - - @Override - public void onActivityDestroyed(Activity activity) { - //System.out.println(activity.getLocalClassName() + " was destroyed"); - LogUtils.d(TAG, activity.getLocalClassName() + " was destroyed"); - } -} diff --git a/positions/src/main/java/cc/winboll/studio/positions/views/HourglassView.java b/positions/src/main/java/cc/winboll/studio/positions/views/HourglassView.java deleted file mode 100644 index 9eb3f95..0000000 --- a/positions/src/main/java/cc/winboll/studio/positions/views/HourglassView.java +++ /dev/null @@ -1,282 +0,0 @@ -package cc.winboll.studio.positions.views; - -/** - * @Author ZhanGSKen&豆包大模型 - * @Date 2025/11/10 08:29 - * @Describe 沙漏计时器控件 - */ -import android.content.Context; -import android.graphics.Color; -import android.graphics.drawable.ClipDrawable; -import android.graphics.drawable.ColorDrawable; -import android.graphics.drawable.Drawable; -import android.graphics.drawable.LayerDrawable; -import android.text.InputFilter; -import android.text.InputType; -import android.util.AttributeSet; -import android.view.Gravity; -import android.view.ViewGroup; -import android.widget.EditText; -import android.widget.LinearLayout; -import android.widget.ProgressBar; -import android.widget.Switch; -import android.widget.TextView; - -/** - * 沙漏视图类(Java 7语法,修复ProgressDrawable和setHeight问题) - */ -public class HourglassView extends LinearLayout { - public static final String TAG = "HourglassView"; - // 数据模型 - private String hourglassId; - private int hour; // 小时 - private int minute; // 分钟 - private boolean isEnabled; // 开关状态 - - // 控件引用 - private EditText etHour; - private EditText etMinute; - private ProgressBar progressBar; - private Switch switchControl; - - // 样式参数 - private int textSize = 16; - private int padding = 8; - private int progressColor = 0xFF2196F3; // 进度条颜色 - private int progressBgColor = 0xFFE0E0E0; // 进度条背景色 - private int textColor = 0xFF333333; - private int editTextWidth = 40; // 输入框宽度(dp) - private int progressHeight = 8; // 进度条高度(dp,新增参数) - - public HourglassView(Context context) { - super(context); - initView(); - } - - public HourglassView(Context context, AttributeSet attrs) { - super(context, attrs); - initView(); - } - - /** - * 初始化视图布局 - */ - private void initView() { - setOrientation(HORIZONTAL); - setGravity(Gravity.CENTER_VERTICAL); - setPadding(dp2px(padding), dp2px(padding), dp2px(padding), dp2px(padding)); - - // 1. 左侧时间输入区域(水平布局) - LinearLayout inputLayout = new LinearLayout(getContext()); - inputLayout.setOrientation(HORIZONTAL); - inputLayout.setGravity(Gravity.CENTER_VERTICAL); - LayoutParams inputParams = new LayoutParams( - ViewGroup.LayoutParams.WRAP_CONTENT, - ViewGroup.LayoutParams.WRAP_CONTENT - ); - inputParams.setMargins(0, 0, dp2px(padding * 2), 0); - addView(inputLayout, inputParams); - - // 小时输入框 - etHour = createNumberEditText(); - etHour.setHint("时"); - etHour.setFilters(new InputFilter[]{new InputFilter.LengthFilter(2)}); - inputLayout.addView(etHour, getEditTextParams()); - - // 分隔符 - TextView divider = new TextView(getContext()); - divider.setText(":"); - divider.setTextSize(textSize); - divider.setTextColor(textColor); - LayoutParams dividerParams = new LayoutParams( - ViewGroup.LayoutParams.WRAP_CONTENT, - ViewGroup.LayoutParams.WRAP_CONTENT - ); - dividerParams.setMargins(dp2px(padding / 2), 0, dp2px(padding / 2), 0); - inputLayout.addView(divider, dividerParams); - - // 分钟输入框 - etMinute = createNumberEditText(); - etMinute.setHint("分"); - etMinute.setFilters(new InputFilter[]{new InputFilter.LengthFilter(2)}); - inputLayout.addView(etMinute, getEditTextParams()); - - // 2. 中间进度条(修复:通过LayoutParams设置高度,替代setHeight) - progressBar = new ProgressBar(getContext(), null, android.R.attr.progressBarStyleHorizontal); - progressBar.setProgressDrawable(createProgressDrawable()); // 传入Drawable类型 - - // 修复核心:用LayoutParams设置进度条高度(兼容低版本) - LayoutParams progressParams = new LayoutParams( - 0, - dp2px(progressHeight), // 直接在布局参数中设置高度(dp转px) - 1.0f - ); - progressParams.setMargins(0, 0, dp2px(padding * 2), 0); - addView(progressBar, progressParams); - - // 3. 右侧开关 - switchControl = new Switch(getContext()); - switchControl.setOnCheckedChangeListener(new Switch.OnCheckedChangeListener() { - @Override - public void onCheckedChanged(android.widget.CompoundButton buttonView, boolean isChecked) { - isEnabled = isChecked; - // 开关状态控制输入框是否可编辑 - etHour.setEnabled(!isChecked); - etMinute.setEnabled(!isChecked); - // 更新进度条(仅在开关开启时生效) - if (isChecked) { - updateProgressBar(); - } - } - }); - addView(switchControl); - - // 初始状态 - isEnabled = false; - etHour.setEnabled(true); - etMinute.setEnabled(true); - } - - /** - * 创建数字输入框 - */ - private EditText createNumberEditText() { - EditText editText = new EditText(getContext()); - editText.setInputType(InputType.TYPE_CLASS_NUMBER); - editText.setTextSize(textSize); - editText.setTextColor(textColor); - editText.setGravity(Gravity.CENTER); - editText.setSingleLine(true); - editText.setBackgroundResource(android.R.drawable.edit_text); // 默认输入框背景 - return editText; - } - - /** - * 获取输入框布局参数 - */ - private LayoutParams getEditTextParams() { - LayoutParams params = new LayoutParams( - dp2px(editTextWidth), - ViewGroup.LayoutParams.WRAP_CONTENT - ); - params.setMargins(0, 0, dp2px(padding), 0); - return params; - } - - /** - * 修复核心:创建ProgressDrawable(返回Drawable类型,而非Paint) - * 用LayerDrawable实现「背景+进度」的双层进度条 - */ - private Drawable createProgressDrawable() { - // 1. 进度条背景(灰色) - ColorDrawable bgDrawable = new ColorDrawable(progressBgColor); - // 2. 进度条前景(主题色) - ColorDrawable progressDrawable = new ColorDrawable(progressColor); - // 3. 用ClipDrawable包裹前景,实现进度裁剪 - ClipDrawable clipDrawable = new ClipDrawable(progressDrawable, Gravity.LEFT, ClipDrawable.HORIZONTAL); - - // 4. 组合成LayerDrawable(顺序:背景在下,进度在上) - Drawable[] layers = new Drawable[]{bgDrawable, clipDrawable}; - LayerDrawable layerDrawable = new LayerDrawable(layers); - - // 5. 设置进度条的层级ID(必须与系统ProgressBar的ID匹配) - layerDrawable.setId(0, android.R.id.background); - layerDrawable.setId(1, android.R.id.progress); - - return layerDrawable; - } - - /** - * 更新进度条(总时间 = 小时*60 + 分钟,单位:分钟) - */ - private void updateProgressBar() { - try { - // 获取输入的时间(为空时默认0) - int inputHour = TextUtils.isEmpty(etHour.getText().toString().trim()) - ? 0 : Integer.parseInt(etHour.getText().toString().trim()); - int inputMinute = TextUtils.isEmpty(etMinute.getText().toString().trim()) - ? 0 : Integer.parseInt(etMinute.getText().toString().trim()); - - // 校验时间合法性(小时0-99,分钟0-59) - inputHour = Math.max(0, Math.min(99, inputHour)); - inputMinute = Math.max(0, Math.min(59, inputMinute)); - - // 计算总分钟数(进度条最大值) - int totalMinutes = inputHour * 60 + inputMinute; - totalMinutes = Math.max(1, totalMinutes); // 最小1分钟,避免进度条无长度 - - // 更新进度条 - progressBar.setMax(totalMinutes); - progressBar.setProgress(totalMinutes); // 初始显示满进度,可根据实际需求修改 - - // 更新数据模型 - this.hour = inputHour; - this.minute = inputMinute; - - } catch (NumberFormatException e) { - // 输入非法时重置进度条 - progressBar.setMax(0); - progressBar.setProgress(0); - } - } - - /** - * dp转px(适配不同设备) - */ - private int dp2px(int dp) { - return (int) (dp * getContext().getResources().getDisplayMetrics().density + 0.5f); - } - - // ------------------- 数据模型 getter/setter ------------------- - public String getHourglassId() { - return hourglassId; - } - - public void setHourglassId(String hourglassId) { - this.hourglassId = hourglassId; - } - - public int getHour() { - return hour; - } - - public void setHour(int hour) { - this.hour = Math.max(0, Math.min(99, hour)); // 限制范围 - etHour.setText(String.valueOf(this.hour)); - } - - public int getMinute() { - return minute; - } - - public void setMinute(int minute) { - this.minute = Math.max(0, Math.min(59, minute)); // 限制范围 - etMinute.setText(String.valueOf(this.minute)); - } - - public boolean isEnabled() { - return isEnabled; - } - - public void setEnabled(boolean enabled) { - isEnabled = enabled; - switchControl.setChecked(enabled); - } - - /** - * 手动更新进度条(外部调用) - */ - public void refreshProgress() { - if (isEnabled) { - updateProgressBar(); - } - } - - // 工具类:判断字符串是否为空(Java7无TextUtils.isEmpty,手动实现) - private static class TextUtils { - public static boolean isEmpty(CharSequence str) { - return str == null || str.length() == 0; - } - } -} - diff --git a/positions/src/main/java/cc/winboll/studio/positions/views/PositionTaskListView.java b/positions/src/main/java/cc/winboll/studio/positions/views/PositionTaskListView.java index f0d147c..b9661b3 100644 --- a/positions/src/main/java/cc/winboll/studio/positions/views/PositionTaskListView.java +++ b/positions/src/main/java/cc/winboll/studio/positions/views/PositionTaskListView.java @@ -33,10 +33,6 @@ import java.util.Calendar; import java.util.Date; import java.util.List; import java.util.Locale; -import cc.winboll.studio.positions.App; -import cc.winboll.studio.positions.AppLevel; -import cc.winboll.studio.positions.activities.WinBoLLActivity; -import cc.winboll.studio.positions.PointLevel; public class PositionTaskListView extends LinearLayout { // 视图模式常量 @@ -384,7 +380,7 @@ public class PositionTaskListView extends LinearLayout { // 步骤3:刷新Adapter(局部刷新+范围通知,避免列表错乱) notifyItemRemoved(position); notifyItemRangeChanged(position, mAdapterData.size()); - + LogUtils.d(TAG, "Adapter已移除任务,刷新列表(位置索引=" + position + ")"); // 步骤4:通知外部(如Activity)任务已更新 @@ -461,7 +457,7 @@ public class PositionTaskListView extends LinearLayout { } }); } - + private String genSelectedTimeText(long timeMillis) { // 2. 格式化时间字符串(Java 7 用 SimpleDateFormat,需处理 ParseException) SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm", Locale.getDefault()); @@ -487,16 +483,6 @@ public class PositionTaskListView extends LinearLayout { final EditText etEditDistance = dialogView.findViewById(R.id.et_edit_distance); Button btnCancel = dialogView.findViewById(R.id.btn_dialog_cancel); Button btnSave = dialogView.findViewById(R.id.btn_dialog_save); - HourglassView hourglassView = dialogView.findViewById(R.id.hourglassView); - if (WinBoLLActivity._mPointLevel == PointLevel.WUKONG) { - hourglassView.setVisibility(View.GONE); - } else if (WinBoLLActivity._mPointLevel == PointLevel.LAOJUN) { - hourglassView.setHourglassId("hourglass_001"); - hourglassView.setHour(1); - hourglassView.setMinute(30); - hourglassView.setEnabled(false); // 开启开关 - } - // 绑定外层对话框内的控件 diff --git a/positions/src/main/res/drawable/ic_positions_plus.png b/positions/src/main/res/drawable/ic_positions_plus.png deleted file mode 100644 index 4155d06..0000000 Binary files a/positions/src/main/res/drawable/ic_positions_plus.png and /dev/null differ diff --git a/positions/src/main/res/layout/dialog_edit_task.xml b/positions/src/main/res/layout/dialog_edit_task.xml index ec00b89..4dd0362 100644 --- a/positions/src/main/res/layout/dialog_edit_task.xml +++ b/positions/src/main/res/layout/dialog_edit_task.xml @@ -65,16 +65,6 @@ - - - - - + + 悟空笔记 - 时空任务 - 开疆扩土 - 返璞归真 - 余力不足 - 辎重难返 diff --git a/positions/src/main/res/values/strings.xml b/positions/src/main/res/values/strings.xml index 3913fe0..495482a 100644 --- a/positions/src/main/res/values/strings.xml +++ b/positions/src/main/res/values/strings.xml @@ -1,8 +1,3 @@ Positions - PositionsPlus - Open APP Plus - Close APP Plus - APP Plus Open Disable - APP Plus Close Disable diff --git a/positions/src/main/res/xml/file_provider.xml b/positions/src/main/res/xml/file_provider.xml deleted file mode 100644 index 802e4cc..0000000 --- a/positions/src/main/res/xml/file_provider.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - diff --git a/positions/src/main/res/xml/shortcutsmain.xml b/positions/src/main/res/xml/shortcutsmain.xml deleted file mode 100644 index 5c0c0b4..0000000 --- a/positions/src/main/res/xml/shortcutsmain.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - - - - - - diff --git a/positions/src/main/res/xml/shortcutsplus.xml b/positions/src/main/res/xml/shortcutsplus.xml deleted file mode 100644 index a38aa91..0000000 --- a/positions/src/main/res/xml/shortcutsplus.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - - - - - -