Compare commits
21 Commits
apputils-v
...
powerbell-
| Author | SHA1 | Date | |
|---|---|---|---|
| 4af10c74e9 | |||
|
|
7883ef93be | ||
|
|
398cb33a58 | ||
|
|
ac1858fba7 | ||
|
|
5a77a5e9e0 | ||
| f943db17e0 | |||
|
|
d7a9cb2a20 | ||
| de34c33706 | |||
|
|
10b8da2e21 | ||
|
|
ca4e4c7feb | ||
| 4108371c20 | |||
|
|
e5c8624d9b | ||
| 561330697b | |||
|
|
f7b2c0d4c0 | ||
|
|
c3978a1e3c | ||
| 5e198d9c68 | |||
|
|
963a3bb7cd | ||
| e9bb789daa | |||
| dbff19e7f4 | |||
|
|
44679d0c8a | ||
| 6656161903 |
@@ -1,8 +1,8 @@
|
|||||||
#Created by .winboll/winboll_app_build.gradle
|
#Created by .winboll/winboll_app_build.gradle
|
||||||
#Mon Sep 22 06:01:08 HKT 2025
|
#Sat Sep 27 21:03:20 HKT 2025
|
||||||
stageCount=8
|
stageCount=10
|
||||||
libraryProject=libappbase
|
libraryProject=libappbase
|
||||||
baseVersion=15.10
|
baseVersion=15.10
|
||||||
publishVersion=15.10.7
|
publishVersion=15.10.9
|
||||||
buildCount=0
|
buildCount=0
|
||||||
baseBetaVersion=15.10.8
|
baseBetaVersion=15.10.10
|
||||||
|
|||||||
@@ -1,35 +0,0 @@
|
|||||||
# APPUtils
|
|
||||||
|
|
||||||
#### 介绍
|
|
||||||
应用开发工具套件类
|
|
||||||
|
|
||||||
#### 软件架构
|
|
||||||
适配安卓应用 [AIDE Pro] 的 Gradle 编译结构。
|
|
||||||
也适配安卓应用 [AndroidIDE] 的 Gradle 编译结构。
|
|
||||||
|
|
||||||
|
|
||||||
#### Gradle 编译说明
|
|
||||||
调试版编译命令 :gradle assembleBetaDebug
|
|
||||||
阶段版编译命令 :git pull && bash .winboll/bashPublishAPKAddTag.sh apputils
|
|
||||||
阶段版类库发布命令 :git pull &&bash .winboll/bashPublishLIBAddTag.sh libapputils
|
|
||||||
|
|
||||||
#### 使用说明
|
|
||||||
|
|
||||||
#### 参与贡献
|
|
||||||
|
|
||||||
1. Fork 本仓库
|
|
||||||
2. 新建 Feat_xxx 分支
|
|
||||||
3. 提交代码 : ZhanGSKen(ZhanGSKen<zhangsken@188.com>)
|
|
||||||
4. 新建 Pull Request
|
|
||||||
|
|
||||||
|
|
||||||
#### 特技
|
|
||||||
|
|
||||||
1. 使用 Readme\_XXX.md 来支持不同的语言,例如 Readme\_en.md, Readme\_zh.md
|
|
||||||
2. Gitee 官方博客 [blog.gitee.com](https://blog.gitee.com)
|
|
||||||
3. 你可以 [https://gitee.com/explore](https://gitee.com/explore) 这个地址来了解 Gitee 上的优秀开源项目
|
|
||||||
4. [GVP](https://gitee.com/gvp) 全称是 Gitee 最有价值开源项目,是综合评定出的优秀开源项目
|
|
||||||
5. Gitee 官方提供的使用手册 [https://gitee.com/help](https://gitee.com/help)
|
|
||||||
6. Gitee 封面人物是一档用来展示 Gitee 会员风采的栏目 [https://gitee.com/gitee-stars/](https://gitee.com/gitee-stars/)
|
|
||||||
|
|
||||||
#### 参考文档
|
|
||||||
@@ -29,7 +29,7 @@ android {
|
|||||||
// versionName 更新后需要手动设置
|
// versionName 更新后需要手动设置
|
||||||
// 项目模块目录的 build.gradle 文件的 stageCount=0
|
// 项目模块目录的 build.gradle 文件的 stageCount=0
|
||||||
// Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
|
// Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
|
||||||
versionName "15.10"
|
versionName "15.8"
|
||||||
if(true) {
|
if(true) {
|
||||||
versionName = genVersionName("${versionName}")
|
versionName = genVersionName("${versionName}")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
#Created by .winboll/winboll_app_build.gradle
|
#Created by .winboll/winboll_app_build.gradle
|
||||||
#Mon Sep 29 00:19:37 HKT 2025
|
#Mon Sep 01 07:56:33 HKT 2025
|
||||||
stageCount=2
|
stageCount=7
|
||||||
libraryProject=libapputils
|
libraryProject=libapputils
|
||||||
baseVersion=15.10
|
baseVersion=15.8
|
||||||
publishVersion=15.10.1
|
publishVersion=15.8.6
|
||||||
buildCount=0
|
buildCount=0
|
||||||
baseBetaVersion=15.10.2
|
baseBetaVersion=15.8.7
|
||||||
|
|||||||
@@ -2,20 +2,14 @@
|
|||||||
<manifest
|
<manifest
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
package="cc.winboll.studio.apputils">
|
package="cc.winboll.studio.apputils">
|
||||||
|
|
||||||
<!-- 读取外部存储权限(Android 10 以下) -->
|
|
||||||
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
|
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:name=".App"
|
android:name=".App"
|
||||||
android:allowBackup="true"
|
android:allowBackup="true"
|
||||||
android:icon="@drawable/ic_winboll"
|
android:icon="@drawable/ic_winboll"
|
||||||
android:label="@string/app_name"
|
android:label="@string/app_name"
|
||||||
android:theme="@style/MyUtilsTheme"
|
android:theme="@style/MyUtilsTheme"
|
||||||
android:supportsRtl="true"
|
android:supportsRtl="true">
|
||||||
android:resizeableActivity="true"
|
|
||||||
android:screenOrientation="unspecified"
|
|
||||||
android:requestLegacyExternalStorage="true">
|
|
||||||
|
|
||||||
<activity
|
<activity
|
||||||
android:name=".MainActivity"
|
android:name=".MainActivity"
|
||||||
@@ -41,8 +35,6 @@
|
|||||||
|
|
||||||
<activity android:name=".QRCodeDecodeActivity"/>
|
<activity android:name=".QRCodeDecodeActivity"/>
|
||||||
|
|
||||||
<activity android:name=".QRGeneratorActivity"/>
|
|
||||||
|
|
||||||
</application>
|
</application>
|
||||||
|
|
||||||
</manifest>
|
</manifest>
|
||||||
@@ -15,11 +15,12 @@ import android.view.MenuItem;
|
|||||||
import android.widget.Toolbar;
|
import android.widget.Toolbar;
|
||||||
import cc.winboll.studio.apputils.R;
|
import cc.winboll.studio.apputils.R;
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
|
import cc.winboll.studio.libappbase.winboll.IWinBoLLActivity;
|
||||||
import cc.winboll.studio.libapputils.views.SimpleWebView;
|
import cc.winboll.studio.libapputils.views.SimpleWebView;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
|
||||||
public class AssetsHtmlActivity extends Activity {
|
public class AssetsHtmlActivity extends WinBoLLActivity implements IWinBoLLActivity {
|
||||||
|
|
||||||
public static final String TAG = "AssetsHtmlActivity";
|
public static final String TAG = "AssetsHtmlActivity";
|
||||||
|
|
||||||
@@ -31,6 +32,16 @@ public class AssetsHtmlActivity extends Activity {
|
|||||||
|
|
||||||
// Assets 文件夹里的 Html 文件的名称
|
// Assets 文件夹里的 Html 文件的名称
|
||||||
String mszHtmlFileName;
|
String mszHtmlFileName;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Activity getActivity() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTag() {
|
||||||
|
return TAG;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onCreateOptionsMenu(Menu menu) {
|
public boolean onCreateOptionsMenu(Menu menu) {
|
||||||
|
|||||||
@@ -15,10 +15,9 @@ import android.widget.Toolbar;
|
|||||||
import cc.winboll.studio.apputils.R;
|
import cc.winboll.studio.apputils.R;
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
import cc.winboll.studio.libappbase.LogView;
|
import cc.winboll.studio.libappbase.LogView;
|
||||||
import cc.winboll.studio.libappbase.ToastUtils;
|
import cc.winboll.studio.libappbase.utils.ToastUtils;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import cc.winboll.studio.libappbase.LogActivity;
|
|
||||||
|
|
||||||
final public class MainActivity extends Activity {
|
final public class MainActivity extends Activity {
|
||||||
|
|
||||||
@@ -27,21 +26,21 @@ final public class MainActivity extends Activity {
|
|||||||
public static final int REQUEST_QRCODEDECODE_ACTIVITY = 0;
|
public static final int REQUEST_QRCODEDECODE_ACTIVITY = 0;
|
||||||
|
|
||||||
Toolbar mToolbar;
|
Toolbar mToolbar;
|
||||||
//LogView mLogView;
|
LogView mLogView;
|
||||||
//
|
//
|
||||||
// @Override
|
// @Override
|
||||||
// public Activity getActivity() {
|
// public Activity getActivity() {
|
||||||
// return this;
|
// return this;
|
||||||
// }
|
// }
|
||||||
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
setContentView(R.layout.activity_main);
|
setContentView(R.layout.activity_main);
|
||||||
|
|
||||||
// mLogView = findViewById(R.id.logview);
|
mLogView = findViewById(R.id.logview);
|
||||||
// mLogView.start();
|
mLogView.start();
|
||||||
|
|
||||||
// 初始化工具栏
|
// 初始化工具栏
|
||||||
mToolbar = findViewById(R.id.toolbar);
|
mToolbar = findViewById(R.id.toolbar);
|
||||||
@@ -146,21 +145,13 @@ final public class MainActivity extends Activity {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void onTestLogActivity(View view) {
|
public void onTestLogActivity(View view) {
|
||||||
/* 分屏代码有效
|
// Intent intent = new Intent(this, LogActivity.class);
|
||||||
// 1. 创建启动 SecondActivity 的 Intent
|
// intent.addFlags(Intent.FLAG_ACTIVITY_NEW_DOCUMENT);
|
||||||
Intent splitIntent = new Intent(MainActivity.this, LogActivity.class);
|
// intent.addFlags(Intent.FLAG_ACTIVITY_MULTIPLE_TASK);
|
||||||
|
// startActivity(intent);
|
||||||
|
|
||||||
// 2. 添加分屏启动必需的两个标志(API 30 兼容)
|
//WinBoLLActivityManager.getInstance().printAvtivityListInfo();
|
||||||
// FLAG_ACTIVITY_LAUNCH_ADJACENT:相邻分屏显示
|
//WinBoLLActivityManager.getInstance(this).startWinBoLLActivity(this, LogActivity.class);
|
||||||
// FLAG_ACTIVITY_NEW_TASK:分屏需要新任务栈
|
|
||||||
splitIntent.addFlags(Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT
|
|
||||||
| Intent.FLAG_ACTIVITY_NEW_TASK);
|
|
||||||
|
|
||||||
// 3. 启动分屏活动(若设备不支持分屏,会默认全屏启动)
|
|
||||||
startActivity(splitIntent);
|
|
||||||
*/
|
|
||||||
|
|
||||||
LogActivity.startLogActivity(this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
@@ -226,9 +217,10 @@ final public class MainActivity extends Activity {
|
|||||||
if (item.getItemId() == R.id.item_exit) {
|
if (item.getItemId() == R.id.item_exit) {
|
||||||
//exit();
|
//exit();
|
||||||
return true;
|
return true;
|
||||||
} else if (item.getItemId() == R.id.item_testqrgeneratoractivity) {
|
// } else if (item.getItemId() == R.id.item_teststringtoqrcodeview) {
|
||||||
Intent intent = new Intent(this, QRGeneratorActivity.class);
|
// Intent intent = new Intent(this, TestStringToQRCodeViewActivity.class);
|
||||||
startActivity(intent);
|
// startActivityForResult(intent, REQUEST_QRCODEDECODE_ACTIVITY);
|
||||||
|
// //WinBoLLActivityManager.getInstance(this).startWinBoLLActivity(this, TestStringToQrCodeViewActivity.class);
|
||||||
} else if (item.getItemId() == R.id.item_testqrcodedecodeactivity) {
|
} else if (item.getItemId() == R.id.item_testqrcodedecodeactivity) {
|
||||||
Intent intent = new Intent(this, QRCodeDecodeActivity.class);
|
Intent intent = new Intent(this, QRCodeDecodeActivity.class);
|
||||||
startActivityForResult(intent, REQUEST_QRCODEDECODE_ACTIVITY);
|
startActivityForResult(intent, REQUEST_QRCODEDECODE_ACTIVITY);
|
||||||
@@ -276,7 +268,7 @@ final public class MainActivity extends Activity {
|
|||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public void onTestAssetsHtmlActivity(View view) {
|
public void onTestAssetsHtmlActivity(View view) {
|
||||||
Intent intent = new Intent(this, AssetsHtmlActivity.class);
|
Intent intent = new Intent(this, AssetsHtmlActivity.class);
|
||||||
intent.putExtra(AssetsHtmlActivity.EXTRA_HTMLFILENAME, "javascript_test.html");
|
intent.putExtra(AssetsHtmlActivity.EXTRA_HTMLFILENAME, "javascript_test.html");
|
||||||
@@ -289,7 +281,7 @@ final public class MainActivity extends Activity {
|
|||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
//mLogView.start();
|
mLogView.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*@Override
|
/*@Override
|
||||||
|
|||||||
@@ -1,323 +1,89 @@
|
|||||||
package cc.winboll.studio.apputils;
|
package cc.winboll.studio.apputils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @Author ZhanGSKen&豆包大模型<zhangsken@qq.com>
|
* @Author ZhanGSKen<zhangsken@qq.com>
|
||||||
* @Date 2025/01/18 10:32:21
|
* @Date 2025/01/18 10:32:21
|
||||||
* @Describe 二维码解码窗口
|
* @Describe 二维码扫码解码窗口
|
||||||
*/
|
*/
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.app.AlertDialog;
|
|
||||||
import android.content.DialogInterface;
|
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
import android.graphics.Bitmap;
|
|
||||||
import android.graphics.BitmapFactory;
|
|
||||||
import android.graphics.Canvas;
|
|
||||||
import android.graphics.ColorMatrix;
|
|
||||||
import android.graphics.ColorMatrixColorFilter;
|
|
||||||
import android.graphics.Paint;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.provider.MediaStore;
|
|
||||||
import android.view.View;
|
|
||||||
import android.widget.Button;
|
|
||||||
import android.widget.ScrollView;
|
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
import android.widget.Toolbar;
|
import android.widget.Toolbar;
|
||||||
import cc.winboll.studio.apputils.R;
|
import cc.winboll.studio.apputils.R;
|
||||||
import com.google.zxing.BarcodeFormat;
|
|
||||||
import com.google.zxing.BinaryBitmap;
|
|
||||||
import com.google.zxing.DecodeHintType;
|
|
||||||
import com.google.zxing.LuminanceSource;
|
|
||||||
import com.google.zxing.MultiFormatReader;
|
|
||||||
import com.google.zxing.NotFoundException;
|
|
||||||
import com.google.zxing.RGBLuminanceSource;
|
|
||||||
import com.google.zxing.Result;
|
|
||||||
import com.google.zxing.ResultPoint;
|
import com.google.zxing.ResultPoint;
|
||||||
import com.google.zxing.common.HybridBinarizer;
|
|
||||||
import com.journeyapps.barcodescanner.BarcodeCallback;
|
import com.journeyapps.barcodescanner.BarcodeCallback;
|
||||||
import com.journeyapps.barcodescanner.BarcodeResult;
|
import com.journeyapps.barcodescanner.BarcodeResult;
|
||||||
import com.journeyapps.barcodescanner.DecoratedBarcodeView;
|
import com.journeyapps.barcodescanner.DecoratedBarcodeView;
|
||||||
import com.journeyapps.barcodescanner.DefaultDecoderFactory;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Hashtable;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class QRCodeDecodeActivity extends Activity {
|
public class QRCodeDecodeActivity extends Activity {
|
||||||
|
|
||||||
public static final String TAG = "QRCodeDecodeActivity";
|
public static final String TAG = "QRCodeDecodeActivity";
|
||||||
|
|
||||||
public static final String EXTRA_RESULT = "EXTRA_RESULT";
|
public static final String EXTRA_RESULT = "EXTRA_RESULT";
|
||||||
private static final int REQUEST_CAMERA_PERMISSION = 1;
|
private static final int REQUEST_CAMERA_PERMISSION = 1;
|
||||||
private static final int REQUEST_PICK_IMAGE = 2;
|
|
||||||
private static final int REQUEST_READ_STORAGE_PERMISSION = 3;
|
|
||||||
// 图片压缩阈值:超过1000px时压缩,避免内存溢出和解码效率低
|
|
||||||
private static final int MAX_BITMAP_SIZE = 1000;
|
|
||||||
|
|
||||||
private TextView resultTextView;
|
TextView resultTextView;
|
||||||
private DecoratedBarcodeView barcodeView;
|
DecoratedBarcodeView barcodeView;
|
||||||
private Button btnDecodeFromAlbum;
|
|
||||||
|
// @Override
|
||||||
|
// public Activity getActivity() {
|
||||||
|
// return this;
|
||||||
|
// }
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
setContentView(R.layout.activity_qrcodedecode);
|
setContentView(R.layout.activity_qrcodedecode);
|
||||||
initToolbar();
|
|
||||||
initViews();
|
|
||||||
checkCameraPermission();
|
|
||||||
setAlbumDecodeClickListener();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void initToolbar() {
|
// 初始化工具栏
|
||||||
Toolbar mToolbar = (Toolbar) findViewById(R.id.toolbar);
|
// Toolbar mToolbar = findViewById(R.id.toolbar);
|
||||||
if (mToolbar != null) {
|
// setActionBar(mToolbar);
|
||||||
setActionBar(mToolbar);
|
|
||||||
if (getActionBar() != null) {
|
|
||||||
getActionBar().setDisplayHomeAsUpEnabled(true);
|
|
||||||
mToolbar.setNavigationOnClickListener(new View.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(View v) {
|
|
||||||
finish();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void initViews() {
|
//resultTextView = findViewById(R.id.activityqrcodedecodeTextView1);
|
||||||
resultTextView = (TextView) findViewById(R.id.activityqrcodedecodeTextView1);
|
barcodeView = findViewById(R.id.activityqrcodedecodeDecoratedBarcodeView1);
|
||||||
barcodeView = (DecoratedBarcodeView) findViewById(R.id.activityqrcodedecodeDecoratedBarcodeView1);
|
// 请求相机权限
|
||||||
btnDecodeFromAlbum = (Button) findViewById(R.id.btn_decode_from_album);
|
// if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.CAMERA)
|
||||||
// 初始化扫码解码器(支持所有常见码制,避免仅支持QR_CODE的局限)
|
// != PackageManager.PERMISSION_GRANTED) {
|
||||||
List<BarcodeFormat> formats = new ArrayList<BarcodeFormat>();
|
// ActivityCompat.requestPermissions(this,
|
||||||
formats.add(BarcodeFormat.QR_CODE);
|
// new String[]{android.Manifest.permission.CAMERA},
|
||||||
formats.add(BarcodeFormat.CODE_128);
|
// REQUEST_CAMERA_PERMISSION);
|
||||||
formats.add(BarcodeFormat.EAN_13);
|
// } else {
|
||||||
barcodeView.getBarcodeView().setDecoderFactory(new DefaultDecoderFactory(formats));
|
// startScanning();
|
||||||
}
|
// }
|
||||||
|
startScanning();
|
||||||
private void setAlbumDecodeClickListener() {
|
|
||||||
btnDecodeFromAlbum.setOnClickListener(new View.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(View v) {
|
|
||||||
if (checkSelfPermission(android.Manifest.permission.READ_EXTERNAL_STORAGE)
|
|
||||||
!= PackageManager.PERMISSION_GRANTED) {
|
|
||||||
requestPermissions(new String[]{android.Manifest.permission.READ_EXTERNAL_STORAGE},
|
|
||||||
REQUEST_READ_STORAGE_PERMISSION);
|
|
||||||
} else {
|
|
||||||
openAlbum();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private void openAlbum() {
|
|
||||||
Intent pickImageIntent = new Intent(Intent.ACTION_PICK);
|
|
||||||
pickImageIntent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
|
|
||||||
startActivityForResult(pickImageIntent, REQUEST_PICK_IMAGE);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
||||||
super.onActivityResult(requestCode, resultCode, data);
|
|
||||||
if (requestCode == REQUEST_PICK_IMAGE && resultCode == RESULT_OK && data != null) {
|
|
||||||
Uri selectedImageUri = data.getData();
|
|
||||||
if (selectedImageUri != null) {
|
|
||||||
try {
|
|
||||||
// 1. 读取图片并压缩(关键优化:避免大图片解码失败)
|
|
||||||
InputStream imageStream = getContentResolver().openInputStream(selectedImageUri);
|
|
||||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
|
||||||
options.inJustDecodeBounds = true; // 先获取图片尺寸,不加载像素
|
|
||||||
BitmapFactory.decodeStream(imageStream, null, options);
|
|
||||||
imageStream.close(); // 关闭流,重新读取
|
|
||||||
|
|
||||||
// 计算压缩比例:超过MAX_BITMAP_SIZE时按比例压缩
|
|
||||||
options.inSampleSize = calculateInSampleSize(options, MAX_BITMAP_SIZE, MAX_BITMAP_SIZE);
|
|
||||||
options.inJustDecodeBounds = false; // 开始加载压缩后的像素
|
|
||||||
imageStream = getContentResolver().openInputStream(selectedImageUri);
|
|
||||||
Bitmap originalBitmap = BitmapFactory.decodeStream(imageStream, null, options);
|
|
||||||
imageStream.close();
|
|
||||||
|
|
||||||
if (originalBitmap == null) {
|
|
||||||
showToast("图片损坏,无法解析");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. 图片预处理:转为灰度图+提高对比度(解决模糊/低对比度图片识别问题)
|
|
||||||
Bitmap processedBitmap = processBitmap(originalBitmap);
|
|
||||||
|
|
||||||
// 3. 解码预处理后的图片
|
|
||||||
String decodeResult = decodeQrFromBitmap(processedBitmap);
|
|
||||||
|
|
||||||
// 4. 结果处理
|
|
||||||
if (decodeResult != null && !decodeResult.isEmpty()) {
|
|
||||||
resultTextView.setText("图片解码结果:" + decodeResult);
|
|
||||||
showDecodeResultDialog(decodeResult);
|
|
||||||
returnResultToPreviousPage(decodeResult);
|
|
||||||
} else {
|
|
||||||
// 尝试直接解码原图(防止预处理过度导致识别失败)
|
|
||||||
String originalResult = decodeQrFromBitmap(originalBitmap);
|
|
||||||
if (originalResult != null && !originalResult.isEmpty()) {
|
|
||||||
resultTextView.setText("图片解码结果:" + originalResult);
|
|
||||||
showDecodeResultDialog(originalResult);
|
|
||||||
returnResultToPreviousPage(originalResult);
|
|
||||||
} else {
|
|
||||||
new AlertDialog.Builder(this)
|
|
||||||
.setTitle("解码失败")
|
|
||||||
.setMessage("图片中未识别到二维码/条码,建议选择清晰、完整的图片")
|
|
||||||
.setPositiveButton("确定", new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
dialog.dismiss();
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 回收Bitmap,避免内存泄漏
|
|
||||||
if (!originalBitmap.isRecycled()) originalBitmap.recycle();
|
|
||||||
if (!processedBitmap.isRecycled() && processedBitmap != originalBitmap) {
|
|
||||||
processedBitmap.recycle();
|
|
||||||
}
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
showToast("图片处理失败:" + e.getMessage());
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
showToast("未选择图片");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 核心优化1:计算图片压缩比例
|
|
||||||
*/
|
|
||||||
private int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
|
|
||||||
final int height = options.outHeight;
|
|
||||||
final int width = options.outWidth;
|
|
||||||
int inSampleSize = 1;
|
|
||||||
|
|
||||||
if (height > reqHeight || width > reqWidth) {
|
|
||||||
final int halfHeight = height / 2;
|
|
||||||
final int halfWidth = width / 2;
|
|
||||||
// 找到最接近reqWidth/reqHeight的压缩比例(2的倍数,保证图片质量)
|
|
||||||
while ((halfHeight / inSampleSize) >= reqHeight
|
|
||||||
&& (halfWidth / inSampleSize) >= reqWidth) {
|
|
||||||
inSampleSize *= 2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return inSampleSize;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 核心优化2:图片预处理(灰度化+提高对比度)
|
|
||||||
* 解决模糊、低亮度、低对比度图片识别率低的问题
|
|
||||||
*/
|
|
||||||
private Bitmap processBitmap(Bitmap bitmap) {
|
|
||||||
int width = bitmap.getWidth();
|
|
||||||
int height = bitmap.getHeight();
|
|
||||||
|
|
||||||
// 创建灰度图
|
|
||||||
Bitmap grayBitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
|
|
||||||
Canvas canvas = new Canvas(grayBitmap);
|
|
||||||
Paint paint = new Paint();
|
|
||||||
|
|
||||||
// 1. 灰度化矩阵
|
|
||||||
ColorMatrix grayMatrix = new ColorMatrix();
|
|
||||||
grayMatrix.setSaturation(0); // 饱和度设为0,转为灰度
|
|
||||||
|
|
||||||
// 2. 提高对比度矩阵(alpha=1.5,亮度=0,可根据需求调整)
|
|
||||||
ColorMatrix contrastMatrix = new ColorMatrix();
|
|
||||||
contrastMatrix.set(new float[]{
|
|
||||||
1.5f, 0, 0, 0, 0, // 红通道对比度
|
|
||||||
0, 1.5f, 0, 0, 0, // 绿通道对比度
|
|
||||||
0, 0, 1.5f, 0, 0, // 蓝通道对比度
|
|
||||||
0, 0, 0, 1f, 0 // alpha通道不变
|
|
||||||
});
|
|
||||||
|
|
||||||
// 合并灰度+对比度矩阵
|
|
||||||
ColorMatrix combinedMatrix = new ColorMatrix();
|
|
||||||
combinedMatrix.postConcat(grayMatrix);
|
|
||||||
combinedMatrix.postConcat(contrastMatrix);
|
|
||||||
|
|
||||||
paint.setColorFilter(new ColorMatrixColorFilter(combinedMatrix));
|
|
||||||
canvas.drawBitmap(bitmap, 0, 0, paint);
|
|
||||||
|
|
||||||
return grayBitmap;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 核心优化3:修复解码参数,支持更多场景
|
|
||||||
*/
|
|
||||||
private String decodeQrFromBitmap(Bitmap bitmap) {
|
|
||||||
if (bitmap == null) return null;
|
|
||||||
|
|
||||||
try {
|
|
||||||
int width = bitmap.getWidth();
|
|
||||||
int height = bitmap.getHeight();
|
|
||||||
int[] pixels = new int[width * height];
|
|
||||||
bitmap.getPixels(pixels, 0, width, 0, 0, width, height);
|
|
||||||
|
|
||||||
// 修复1:使用RGBLuminanceSource,避免YUV格式导致的颜色偏差
|
|
||||||
LuminanceSource source = new RGBLuminanceSource(width, height, pixels);
|
|
||||||
BinaryBitmap binaryBitmap = new BinaryBitmap(new HybridBinarizer(source));
|
|
||||||
|
|
||||||
// 修复2:完善解码参数,解决模糊、变形二维码识别问题
|
|
||||||
Hashtable<DecodeHintType, Object> hints = new Hashtable<DecodeHintType, Object>();
|
|
||||||
// 支持所有常见码制(不仅限于QR_CODE)
|
|
||||||
List<BarcodeFormat> formats = new ArrayList<BarcodeFormat>();
|
|
||||||
formats.add(BarcodeFormat.QR_CODE);
|
|
||||||
formats.add(BarcodeFormat.CODE_128);
|
|
||||||
formats.add(BarcodeFormat.EAN_8);
|
|
||||||
formats.add(BarcodeFormat.EAN_13);
|
|
||||||
hints.put(DecodeHintType.POSSIBLE_FORMATS, formats);
|
|
||||||
// 字符编码:支持中文等多语言
|
|
||||||
hints.put(DecodeHintType.CHARACTER_SET, "UTF-8");
|
|
||||||
// 容错模式:允许二维码有一定损坏(关键!解决轻微变形/污染的二维码)
|
|
||||||
hints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE);
|
|
||||||
// 不使用纯条码模式,兼容带logo的二维码
|
|
||||||
hints.put(DecodeHintType.PURE_BARCODE, Boolean.FALSE);
|
|
||||||
|
|
||||||
MultiFormatReader reader = new MultiFormatReader();
|
|
||||||
reader.setHints(hints);
|
|
||||||
Result result = reader.decode(binaryBitmap);
|
|
||||||
return result.getText();
|
|
||||||
|
|
||||||
} catch (NotFoundException e) {
|
|
||||||
// 正常未识别到,不打印异常(避免日志冗余)
|
|
||||||
return null;
|
|
||||||
} catch (Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ==================== 原有逻辑(不变) ====================
|
|
||||||
private void checkCameraPermission() {
|
|
||||||
if (checkSelfPermission(android.Manifest.permission.CAMERA)
|
|
||||||
!= PackageManager.PERMISSION_GRANTED) {
|
|
||||||
requestPermissions(new String[]{android.Manifest.permission.CAMERA},
|
|
||||||
REQUEST_CAMERA_PERMISSION);
|
|
||||||
} else {
|
|
||||||
startScanning();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void startScanning() {
|
private void startScanning() {
|
||||||
|
barcodeView.getBarcodeView().setDecoderFactory(null);
|
||||||
barcodeView.decodeContinuous(barcodeCallback);
|
barcodeView.decodeContinuous(barcodeCallback);
|
||||||
}
|
}
|
||||||
|
|
||||||
private BarcodeCallback barcodeCallback = new BarcodeCallback() {
|
@Override
|
||||||
|
public void onRequestPermissionsResult(int requestCode,
|
||||||
|
String[] permissions, int[] grantResults) {
|
||||||
|
if (requestCode == REQUEST_CAMERA_PERMISSION) {
|
||||||
|
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
|
||||||
|
startScanning();
|
||||||
|
} else {
|
||||||
|
// 权限被拒绝的处理
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BarcodeCallback barcodeCallback = new BarcodeCallback(){
|
||||||
@Override
|
@Override
|
||||||
public void barcodeResult(BarcodeResult result) {
|
public void barcodeResult(BarcodeResult result) {
|
||||||
if (result != null && result.getText() != null) {
|
if (result.getText() != null) {
|
||||||
|
//Toast.makeText(MainActivity.this, "Scanned: " + result.getText(), Toast.LENGTH_SHORT).show();
|
||||||
|
//ToastUtils.show("Scanned: " + result.getText());
|
||||||
barcodeView.pause();
|
barcodeView.pause();
|
||||||
String decodeResult = result.getText();
|
Intent intent = new Intent();
|
||||||
resultTextView.setText("扫码结果:" + decodeResult);
|
intent.putExtra(EXTRA_RESULT, result.getText());
|
||||||
showDecodeResultDialog(decodeResult);
|
setResult(RESULT_OK, intent);
|
||||||
returnResultToPreviousPage(decodeResult);
|
finish();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -326,111 +92,16 @@ public class QRCodeDecodeActivity extends Activity {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
private void showDecodeResultDialog(String result) {
|
|
||||||
ScrollView scrollView = new ScrollView(this);
|
|
||||||
scrollView.setPadding(dip2px(16), dip2px(16), dip2px(16), dip2px(16));
|
|
||||||
|
|
||||||
TextView dialogTv = new TextView(this);
|
|
||||||
dialogTv.setTextSize(16);
|
|
||||||
dialogTv.setTextColor(getResources().getColor(android.R.color.black));
|
|
||||||
dialogTv.setText(result);
|
|
||||||
scrollView.addView(dialogTv);
|
|
||||||
|
|
||||||
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
|
||||||
builder.setTitle("解码结果");
|
|
||||||
builder.setView(scrollView);
|
|
||||||
builder.setPositiveButton("确定", new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
dialog.dismiss();
|
|
||||||
barcodeView.resume();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
builder.setCancelable(false);
|
|
||||||
builder.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void returnResultToPreviousPage(String result) {
|
|
||||||
Intent intent = new Intent();
|
|
||||||
intent.putExtra(EXTRA_RESULT, result);
|
|
||||||
setResult(RESULT_OK, intent);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
|
|
||||||
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
|
|
||||||
if (requestCode == REQUEST_CAMERA_PERMISSION) {
|
|
||||||
if (grantResults != null && grantResults.length > 0) {
|
|
||||||
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
|
|
||||||
startScanning();
|
|
||||||
} else {
|
|
||||||
new AlertDialog.Builder(this)
|
|
||||||
.setTitle("权限申请")
|
|
||||||
.setMessage("扫码需要相机权限,请在设置中开启")
|
|
||||||
.setPositiveButton("确定", new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
dialog.dismiss();
|
|
||||||
finish();
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.setCancelable(false)
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (requestCode == REQUEST_READ_STORAGE_PERMISSION) {
|
|
||||||
if (grantResults != null && grantResults.length > 0) {
|
|
||||||
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
|
|
||||||
openAlbum();
|
|
||||||
} else {
|
|
||||||
new AlertDialog.Builder(this)
|
|
||||||
.setTitle("权限申请")
|
|
||||||
.setMessage("从相册解码需要存储权限,请在设置中开启")
|
|
||||||
.setPositiveButton("确定", new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
dialog.dismiss();
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.setCancelable(false)
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
if (barcodeView != null) {
|
barcodeView.resume();
|
||||||
barcodeView.resume();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onPause() {
|
protected void onPause() {
|
||||||
super.onPause();
|
super.onPause();
|
||||||
if (barcodeView != null) {
|
barcodeView.pause();
|
||||||
barcodeView.pause();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private int dip2px(float dpValue) {
|
|
||||||
final float scale = getResources().getDisplayMetrics().density;
|
|
||||||
return (int) (dpValue * scale + 0.5f);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onOptionsItemSelected(android.view.MenuItem item) {
|
|
||||||
if (item.getItemId() == android.R.id.home) {
|
|
||||||
finish();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return super.onOptionsItemSelected(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void showToast(String message) {
|
|
||||||
android.widget.Toast.makeText(this, message, android.widget.Toast.LENGTH_SHORT).show();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,97 +0,0 @@
|
|||||||
package cc.winboll.studio.apputils;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @Author ZhanGSKen&豆包大模型<zhangsken@qq.com>
|
|
||||||
* @Date 2025/09/22 07:09
|
|
||||||
* @Describe 二维码生成窗口
|
|
||||||
*/
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.graphics.Bitmap;
|
|
||||||
import android.os.Bundle;
|
|
||||||
import android.view.View;
|
|
||||||
import android.widget.Button;
|
|
||||||
import android.widget.EditText;
|
|
||||||
import android.widget.ImageView;
|
|
||||||
import android.widget.Toast;
|
|
||||||
import com.google.zxing.BarcodeFormat;
|
|
||||||
import com.google.zxing.WriterException;
|
|
||||||
import com.journeyapps.barcodescanner.BarcodeEncoder;
|
|
||||||
|
|
||||||
public class QRGeneratorActivity extends Activity {
|
|
||||||
public static final String TAG = "QrGeneratorActivity";
|
|
||||||
|
|
||||||
// 控件引用
|
|
||||||
private EditText etInputText;
|
|
||||||
private ImageView ivQrPreview;
|
|
||||||
private Button btnGenerateQr;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
|
||||||
super.onCreate(savedInstanceState);
|
|
||||||
setContentView(R.layout.activity_qrgenerator);
|
|
||||||
|
|
||||||
// 初始化控件
|
|
||||||
initViews();
|
|
||||||
// 设置按钮点击事件
|
|
||||||
setGenerateClickListener();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 初始化布局控件
|
|
||||||
*/
|
|
||||||
private void initViews() {
|
|
||||||
etInputText = findViewById(R.id.et_input_text);
|
|
||||||
ivQrPreview = findViewById(R.id.iv_qr_preview);
|
|
||||||
btnGenerateQr = findViewById(R.id.btn_generate_qr);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 设置生成按钮点击事件:获取输入文字 → 生成二维码 → 显示到 ImageView
|
|
||||||
*/
|
|
||||||
private void setGenerateClickListener() {
|
|
||||||
btnGenerateQr.setOnClickListener(new View.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(View v) {
|
|
||||||
// 1. 获取输入框文字(去除前后空格)
|
|
||||||
String inputText = etInputText.getText().toString().trim();
|
|
||||||
|
|
||||||
// 2. 空输入判断
|
|
||||||
if (inputText.isEmpty()) {
|
|
||||||
Toast.makeText(QRGeneratorActivity.this, "请先输入要生成二维码的文字", Toast.LENGTH_SHORT).show();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 3. 生成二维码 Bitmap(宽高 500px,可调整)
|
|
||||||
Bitmap qrBitmap = generateQrCodeBitmap(inputText, 500, 500);
|
|
||||||
|
|
||||||
// 4. 显示二维码到 ImageView
|
|
||||||
if (qrBitmap != null) {
|
|
||||||
ivQrPreview.setImageBitmap(qrBitmap);
|
|
||||||
} else {
|
|
||||||
Toast.makeText(QRGeneratorActivity.this, "二维码生成失败,请重试", Toast.LENGTH_SHORT).show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 核心方法:生成二维码 Bitmap
|
|
||||||
* @param content 二维码内容(输入的文字)
|
|
||||||
* @param width 二维码宽度(px)
|
|
||||||
* @param height 二维码高度(px)
|
|
||||||
* @return 生成的二维码 Bitmap,失败返回 null
|
|
||||||
*/
|
|
||||||
private Bitmap generateQrCodeBitmap(String content, int width, int height) {
|
|
||||||
try {
|
|
||||||
// 初始化二维码编码器(指定格式为 QR_CODE)
|
|
||||||
BarcodeEncoder encoder = new BarcodeEncoder();
|
|
||||||
// 生成二维码 Bitmap(参数:内容、格式、宽、高)
|
|
||||||
return encoder.encodeBitmap(content, BarcodeFormat.QR_CODE, width, height);
|
|
||||||
} catch (WriterException e) {
|
|
||||||
// 生成失败(如内容过长、宽高非法),打印异常信息
|
|
||||||
e.printStackTrace();
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -6,9 +6,19 @@ package cc.winboll.studio.apputils;
|
|||||||
* @Describe WinBoLLActivity
|
* @Describe WinBoLLActivity
|
||||||
*/
|
*/
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
|
import cc.winboll.studio.libappbase.winboll.IWinBoLLActivity;
|
||||||
|
|
||||||
public class WinBoLLActivity extends Activity {
|
public class WinBoLLActivity extends Activity implements IWinBoLLActivity {
|
||||||
|
|
||||||
public static final String TAG = "WinBoLLActivity";
|
public static final String TAG = "WinBoLLActivity";
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Activity getActivity() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTag() {
|
||||||
|
return TAG;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -54,6 +54,13 @@
|
|||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
|
<cc.winboll.studio.libappbase.LogView
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:text="Button"
|
||||||
|
android:id="@+id/logview"
|
||||||
|
android:layout_weight="1.0"/>
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|||||||
@@ -1,43 +1,20 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<LinearLayout
|
<LinearLayout
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
android:orientation="vertical"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent">
|
||||||
android:orientation="vertical">
|
|
||||||
|
|
||||||
<!-- 工具栏(原有) -->
|
|
||||||
<Toolbar
|
|
||||||
android:id="@+id/toolbar"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"/>
|
|
||||||
|
|
||||||
<!-- 扫码控件(原有) -->
|
|
||||||
<com.journeyapps.barcodescanner.DecoratedBarcodeView
|
|
||||||
android:id="@+id/activityqrcodedecodeDecoratedBarcodeView1"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="0dp"
|
|
||||||
android:layout_weight="1"/>
|
|
||||||
|
|
||||||
<!-- 新增:从相册解码按钮 -->
|
|
||||||
<Button
|
|
||||||
android:id="@+id/btn_decode_from_album"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:text="从相册选图解码"
|
|
||||||
android:textSize="16sp"
|
|
||||||
android:textColor="@android:color/white"
|
|
||||||
android:background="#2196F3"
|
|
||||||
android:paddingVertical="12dp"
|
|
||||||
android:layout_margin="16dp"/>
|
|
||||||
|
|
||||||
<!-- 结果显示TextView(原有) -->
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/activityqrcodedecodeTextView1"
|
android:layout_width="wrap_content"
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:padding="16dp"
|
android:text="QRCodeDecodeActivity"/>
|
||||||
android:text="等待扫码或选择图片..."
|
|
||||||
android:textSize="16sp"/>
|
<com.journeyapps.barcodescanner.DecoratedBarcodeView
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/activityqrcodedecodeDecoratedBarcodeView1"/>
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<LinearLayout
|
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:orientation="vertical"
|
|
||||||
android:padding="20dp"
|
|
||||||
android:gravity="top|center_horizontal">
|
|
||||||
|
|
||||||
<!-- 文字输入框 -->
|
|
||||||
<EditText
|
|
||||||
android:id="@+id/et_input_text"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:hint="请输入要生成二维码的文字"
|
|
||||||
android:inputType="textMultiLine"
|
|
||||||
android:minLines="3"
|
|
||||||
android:maxLines="5"
|
|
||||||
android:padding="12dp"
|
|
||||||
android:background="@android:drawable/editbox_background_normal"
|
|
||||||
android:layout_marginBottom="16dp"/>
|
|
||||||
|
|
||||||
<!-- 生成二维码按钮 -->
|
|
||||||
<Button
|
|
||||||
android:id="@+id/btn_generate_qr"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:text="生成二维码"
|
|
||||||
android:textSize="16sp"
|
|
||||||
android:background="#4CAF50"
|
|
||||||
android:textColor="@android:color/white"
|
|
||||||
android:paddingVertical="12dp"
|
|
||||||
android:layout_marginBottom="24dp"/>
|
|
||||||
|
|
||||||
<!-- 二维码预览图片 -->
|
|
||||||
<ImageView
|
|
||||||
android:id="@+id/iv_qr_preview"
|
|
||||||
android:layout_width="280dp"
|
|
||||||
android:layout_height="280dp"
|
|
||||||
android:scaleType="fitCenter"
|
|
||||||
android:background="@android:drawable/dialog_holo_light_frame"
|
|
||||||
android:contentDescription="二维码预览"/>
|
|
||||||
|
|
||||||
</LinearLayout>
|
|
||||||
|
|
||||||
@@ -2,8 +2,8 @@
|
|||||||
<menu xmlns:android="http://schemas.android.com/apk/res/android"
|
<menu xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto">
|
xmlns:app="http://schemas.android.com/apk/res-auto">
|
||||||
<item
|
<item
|
||||||
android:id="@+id/item_testqrgeneratoractivity"
|
android:id="@+id/item_teststringtoqrcodeview"
|
||||||
android:title="TestQRGeneratorActivity"/>
|
android:title="TestStringToQRCodeViewActivity"/>
|
||||||
<item
|
<item
|
||||||
android:id="@+id/item_testqrcodedecodeactivity"
|
android:id="@+id/item_testqrcodedecodeactivity"
|
||||||
android:title="TestQRCodeDecodeActivity"/>
|
android:title="TestQRCodeDecodeActivity"/>
|
||||||
|
|||||||
21
build.gradle
21
build.gradle
@@ -5,6 +5,16 @@ buildscript {
|
|||||||
// 设置本地Maven仓库路径
|
// 设置本地Maven仓库路径
|
||||||
url 'file:///sdcard/.m2/repository/'
|
url 'file:///sdcard/.m2/repository/'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//米盟通过maven接入时,要做如下配置
|
||||||
|
maven {
|
||||||
|
url "https://repos.xiaomi.com/maven"
|
||||||
|
credentials {
|
||||||
|
username 'mimo-developer'
|
||||||
|
password 'AKCp8ih1PFG9tV8qaLyws67dLGZi8udFM39SfsHgihN15cgsiRvHuxj8JzFmuZjaViVeNawaA'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Nexus Maven 库地址
|
// Nexus Maven 库地址
|
||||||
// "WinBoLL Release"
|
// "WinBoLL Release"
|
||||||
maven { url "https://nexus.winboll.cc/repository/maven-public/" }
|
maven { url "https://nexus.winboll.cc/repository/maven-public/" }
|
||||||
@@ -40,7 +50,16 @@ allprojects {
|
|||||||
// 设置本地Maven仓库路径
|
// 设置本地Maven仓库路径
|
||||||
url 'file:///sdcard/.m2/repository/'
|
url 'file:///sdcard/.m2/repository/'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//米盟通过maven接入时,要做如下配置
|
||||||
|
maven {
|
||||||
|
url "https://repos.xiaomi.com/maven"
|
||||||
|
credentials {
|
||||||
|
username 'mimo-developer'
|
||||||
|
password 'AKCp8ih1PFG9tV8qaLyws67dLGZi8udFM39SfsHgihN15cgsiRvHuxj8JzFmuZjaViVeNawaA'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Nexus Maven 库地址
|
// Nexus Maven 库地址
|
||||||
// "WinBoLL Release"
|
// "WinBoLL Release"
|
||||||
maven { url "https://nexus.winboll.cc/repository/maven-public/" }
|
maven { url "https://nexus.winboll.cc/repository/maven-public/" }
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
#Created by .winboll/winboll_app_build.gradle
|
#Created by .winboll/winboll_app_build.gradle
|
||||||
#Mon Sep 22 02:57:06 HKT 2025
|
#Sat Sep 27 21:03:08 HKT 2025
|
||||||
stageCount=8
|
stageCount=10
|
||||||
libraryProject=libappbase
|
libraryProject=libappbase
|
||||||
baseVersion=15.10
|
baseVersion=15.10
|
||||||
publishVersion=15.10.7
|
publishVersion=15.10.9
|
||||||
buildCount=0
|
buildCount=0
|
||||||
baseBetaVersion=15.10.8
|
baseBetaVersion=15.10.10
|
||||||
|
|||||||
@@ -22,12 +22,12 @@ public class GlobalApplication extends Application {
|
|||||||
GlobalApplication.isDebuging = isDebuging;
|
GlobalApplication.isDebuging = isDebuging;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void saveDebugStatus(GlobalApplication application) {
|
public static void saveDebugStatus(Context context) {
|
||||||
APPModel.saveBeanToFile(application.getAPPModelFilePath(application), new APPModel(GlobalApplication.isDebuging));
|
APPModel.saveBeanToFile(getAPPModelFilePath(context), new APPModel(GlobalApplication.isDebuging));
|
||||||
}
|
}
|
||||||
|
|
||||||
static String getAPPModelFilePath(GlobalApplication application) {
|
static String getAPPModelFilePath(Context context) {
|
||||||
return application.getDataDir().getPath() + "/APPModel.json";
|
return context.getDataDir().getPath() + "/APPModel.json";
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean isDebuging() {
|
public static boolean isDebuging() {
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ android {
|
|||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
api fileTree(dir: 'libs', include: ['*.jar'])
|
api fileTree(dir: 'libs', include: ['*.jar'])
|
||||||
api 'cc.winboll.studio:libappbase:15.10.7'
|
api 'cc.winboll.studio:libappbase:15.9.5'
|
||||||
|
|
||||||
// 二维码类库
|
// 二维码类库
|
||||||
api 'com.google.zxing:core:3.4.1'
|
api 'com.google.zxing:core:3.4.1'
|
||||||
@@ -32,8 +32,6 @@ dependencies {
|
|||||||
|
|
||||||
// Html 解析
|
// Html 解析
|
||||||
api 'org.jsoup:jsoup:1.13.1'
|
api 'org.jsoup:jsoup:1.13.1'
|
||||||
|
|
||||||
api 'com.google.code.gson:gson:2.10.1'
|
|
||||||
|
|
||||||
// SSH
|
// SSH
|
||||||
//api 'com.jcraft:jsch:0.1.55'
|
//api 'com.jcraft:jsch:0.1.55'
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
#Created by .winboll/winboll_app_build.gradle
|
#Created by .winboll/winboll_app_build.gradle
|
||||||
#Mon Sep 29 00:19:37 HKT 2025
|
#Mon Sep 01 07:56:11 HKT 2025
|
||||||
stageCount=2
|
stageCount=7
|
||||||
libraryProject=libapputils
|
libraryProject=libapputils
|
||||||
baseVersion=15.10
|
baseVersion=15.8
|
||||||
publishVersion=15.10.1
|
publishVersion=15.8.6
|
||||||
buildCount=0
|
buildCount=0
|
||||||
baseBetaVersion=15.10.2
|
baseBetaVersion=15.8.7
|
||||||
|
|||||||
@@ -1,29 +0,0 @@
|
|||||||
package cc.winboll.studio.libapputils.utils;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @Author ZhanGSKen<zhangsken@qq.com>
|
|
||||||
* @Date 2025/02/15 20:05:03
|
|
||||||
* @Describe AppUtils
|
|
||||||
*/
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.pm.ApplicationInfo;
|
|
||||||
import android.content.pm.PackageManager;
|
|
||||||
import android.content.pm.PackageManager.NameNotFoundException;
|
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
|
||||||
|
|
||||||
public class AppUtils {
|
|
||||||
|
|
||||||
public static final String TAG = "AppUtils";
|
|
||||||
|
|
||||||
public static String getAppNameByPackageName(Context context, String packageName) {
|
|
||||||
PackageManager packageManager = context.getPackageManager();
|
|
||||||
try {
|
|
||||||
ApplicationInfo applicationInfo = packageManager.getApplicationInfo(packageName, 0);
|
|
||||||
return (String) packageManager.getApplicationLabel(applicationInfo);
|
|
||||||
} catch (NameNotFoundException e) {
|
|
||||||
LogUtils.d(TAG, e, Thread.currentThread().getStackTrace());
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -9,16 +9,10 @@ import android.content.Context;
|
|||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.res.AssetManager;
|
import android.content.res.AssetManager;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.support.v4.content.FileProvider;
|
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.BufferedWriter;
|
|
||||||
import java.io.ByteArrayOutputStream;
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileInputStream;
|
import java.io.FileInputStream;
|
||||||
import java.io.FileOutputStream;
|
import java.io.FileOutputStream;
|
||||||
import java.io.FileReader;
|
|
||||||
import java.io.FileWriter;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
@@ -28,6 +22,7 @@ import java.nio.charset.StandardCharsets;
|
|||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.nio.file.Paths;
|
import java.nio.file.Paths;
|
||||||
|
import android.support.v4.content.FileProvider;
|
||||||
|
|
||||||
public class FileUtils {
|
public class FileUtils {
|
||||||
|
|
||||||
@@ -102,6 +97,36 @@ public class FileUtils {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 把字符串写入文件,指定 UTF-8 编码
|
||||||
|
//
|
||||||
|
public static void writeStringToFile(String szFilePath, String szContent) throws IOException {
|
||||||
|
File file = new File(szFilePath);
|
||||||
|
if (!file.getParentFile().exists()) {
|
||||||
|
file.getParentFile().mkdirs();
|
||||||
|
}
|
||||||
|
FileOutputStream outputStream = new FileOutputStream(file);
|
||||||
|
OutputStreamWriter writer = new OutputStreamWriter(outputStream, StandardCharsets.UTF_8);
|
||||||
|
writer.write(szContent);
|
||||||
|
writer.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 读取文件到字符串,指定 UTF-8 编码
|
||||||
|
//
|
||||||
|
public static String readStringFromFile(String szFilePath) throws IOException {
|
||||||
|
File file = new File(szFilePath);
|
||||||
|
FileInputStream inputStream = new FileInputStream(file);
|
||||||
|
InputStreamReader reader = new InputStreamReader(inputStream, StandardCharsets.UTF_8);
|
||||||
|
StringBuilder content = new StringBuilder();
|
||||||
|
int character;
|
||||||
|
while ((character = reader.read()) != -1) {
|
||||||
|
content.append((char) character);
|
||||||
|
}
|
||||||
|
reader.close();
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
|
||||||
public static boolean copyFile(File srcFile, File dstFile) {
|
public static boolean copyFile(File srcFile, File dstFile) {
|
||||||
if (!srcFile.exists()) {
|
if (!srcFile.exists()) {
|
||||||
LogUtils.d(TAG, "The original file does not exist.");
|
LogUtils.d(TAG, "The original file does not exist.");
|
||||||
@@ -129,113 +154,4 @@ public class FileUtils {
|
|||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 读取文件为字节数组(Java 7 语法)
|
|
||||||
*/
|
|
||||||
public static byte[] readByteArrayFromFile(String filePath) {
|
|
||||||
FileInputStream fis = null;
|
|
||||||
ByteArrayOutputStream bos = null;
|
|
||||||
try {
|
|
||||||
fis = new FileInputStream(filePath);
|
|
||||||
bos = new ByteArrayOutputStream();
|
|
||||||
byte[] buffer = new byte[4096];
|
|
||||||
int bytesRead;
|
|
||||||
while ((bytesRead = fis.read(buffer)) != -1) {
|
|
||||||
bos.write(buffer, 0, bytesRead);
|
|
||||||
}
|
|
||||||
return bos.toByteArray();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
return null;
|
|
||||||
} finally {
|
|
||||||
// 手动关闭流(Java 7 不支持 try-with-resources)
|
|
||||||
if (fis != null) {
|
|
||||||
try {
|
|
||||||
fis.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (bos != null) {
|
|
||||||
try {
|
|
||||||
bos.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 写入字节数组到文件(Java 7 语法)
|
|
||||||
*/
|
|
||||||
public static boolean writeByteArrayToFile(byte[] data, String filePath) {
|
|
||||||
FileOutputStream fos = null;
|
|
||||||
try {
|
|
||||||
fos = new FileOutputStream(filePath);
|
|
||||||
fos.write(data);
|
|
||||||
return true;
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
return false;
|
|
||||||
} finally {
|
|
||||||
if (fos != null) {
|
|
||||||
try {
|
|
||||||
fos.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String readStringFromFile(String filePath) {
|
|
||||||
BufferedReader reader = null;
|
|
||||||
try {
|
|
||||||
reader = new BufferedReader(new FileReader(filePath));
|
|
||||||
StringBuilder content = new StringBuilder();
|
|
||||||
String line;
|
|
||||||
while ((line = reader.readLine()) != null) {
|
|
||||||
content.append(line).append(System.getProperty("line.separator"));
|
|
||||||
}
|
|
||||||
// 去除最后一个换行符(可选)
|
|
||||||
if (content.length() > 0) {
|
|
||||||
content.deleteCharAt(content.length() - 1);
|
|
||||||
}
|
|
||||||
return content.toString();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
return null;
|
|
||||||
} finally {
|
|
||||||
if (reader != null) {
|
|
||||||
try {
|
|
||||||
reader.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean writeStringToFile(String content, String filePath, boolean append) {
|
|
||||||
BufferedWriter writer = null;
|
|
||||||
try {
|
|
||||||
writer = new BufferedWriter(new FileWriter(filePath, append));
|
|
||||||
writer.write(content);
|
|
||||||
return true;
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
return false;
|
|
||||||
} finally {
|
|
||||||
if (writer != null) {
|
|
||||||
try {
|
|
||||||
writer.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,18 +18,22 @@ def genVersionName(def versionName){
|
|||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
compileSdkVersion 32
|
|
||||||
buildToolsVersion "32.0.0"
|
// 1. compileSdkVersion:必须 ≥ targetSdkVersion,建议直接等于 targetSdkVersion(30)
|
||||||
|
compileSdkVersion 30
|
||||||
|
|
||||||
|
// 2. buildToolsVersion:需匹配 compileSdkVersion,建议使用 30.x.x 最新稳定版(无需高于 compileSdkVersion)
|
||||||
|
buildToolsVersion "30.0.3" // 这是 30 对应的最新稳定版,避免使用 beta 版
|
||||||
|
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
applicationId "cc.winboll.studio.powerbell"
|
applicationId "cc.winboll.studio.powerbell"
|
||||||
minSdkVersion 24
|
minSdkVersion 23
|
||||||
targetSdkVersion 30
|
targetSdkVersion 30
|
||||||
versionCode 6
|
versionCode 6
|
||||||
// versionName 更新后需要手动设置
|
// versionName 更新后需要手动设置
|
||||||
// .winboll/winbollBuildProps.properties 文件的 stageCount=0
|
// .winboll/winbollBuildProps.properties 文件的 stageCount=0
|
||||||
// Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
|
// Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
|
||||||
versionName "15.4"
|
versionName "15.11"
|
||||||
if(true) {
|
if(true) {
|
||||||
versionName = genVersionName("${versionName}")
|
versionName = genVersionName("${versionName}")
|
||||||
}
|
}
|
||||||
@@ -41,17 +45,41 @@ android {
|
|||||||
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
|
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 米盟
|
||||||
|
packagingOptions {
|
||||||
|
doNotStrip "*/*/libmimo_1011.so"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
api fileTree(dir: 'libs', include: ['*.jar'])
|
// 米盟
|
||||||
api 'cc.winboll.studio:libaes:15.9.3'
|
implementation 'com.miui.zeus:mimo-ad-sdk:5.3.+'//请使用最新版sdk
|
||||||
api 'cc.winboll.studio:libapputils:15.8.5'
|
//注意:以下5个库必须要引入
|
||||||
api 'cc.winboll.studio:libappbase:15.9.5'
|
//implementation 'androidx.appcompat:appcompat:1.4.1'
|
||||||
|
implementation 'androidx.recyclerview:recyclerview:1.0.0'
|
||||||
// 吐司提示库
|
implementation 'com.google.code.gson:gson:2.8.5'
|
||||||
api 'com.github.getActivity:ToastUtils:10.5'
|
implementation 'com.github.bumptech.glide:glide:4.9.0'
|
||||||
// 应用介绍页类库
|
//annotationProcessor 'com.github.bumptech.glide:compiler:4.9.0'
|
||||||
|
|
||||||
|
|
||||||
|
//--------------aar依赖-----------
|
||||||
|
/*implementation(name: 'mimo-ad-sdk', ext: 'aar')
|
||||||
|
//--------------Maven依赖-----------
|
||||||
|
//implementation 'com.miui.zeus:mimo-ad-sdk:5.3.2'
|
||||||
|
|
||||||
|
//注意:以下5个库必须要引入
|
||||||
|
implementation "androidx.appcompat:appcompat:1.2.0"
|
||||||
|
implementation 'com.google.code.gson:gson:2.8.5'
|
||||||
|
implementation 'androidx.recyclerview:recyclerview:1.0.0'
|
||||||
|
implementation 'com.github.bumptech.glide:glide:4.9.0'
|
||||||
|
//annotationProcessor 'com.github.bumptech.glide:compiler:4.9.0'
|
||||||
|
|
||||||
|
implementation "androidx.navigation:navigation-ui:2.3.5"
|
||||||
|
implementation 'androidx.multidex:multidex:2.0.1'
|
||||||
|
//debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.9.1'
|
||||||
|
*/
|
||||||
|
// 应用介绍页类库
|
||||||
api 'io.github.medyo:android-about-page:2.0.0'
|
api 'io.github.medyo:android-about-page:2.0.0'
|
||||||
// SSH
|
// SSH
|
||||||
api 'com.jcraft:jsch:0.1.55'
|
api 'com.jcraft:jsch:0.1.55'
|
||||||
@@ -71,23 +99,9 @@ dependencies {
|
|||||||
//api 'androidx.vectordrawable:vectordrawable-animated:1.1.0'
|
//api 'androidx.vectordrawable:vectordrawable-animated:1.1.0'
|
||||||
//api 'androidx.fragment:fragment:1.1.0'
|
//api 'androidx.fragment:fragment:1.1.0'
|
||||||
|
|
||||||
|
implementation 'cc.winboll.studio:libaes:15.11.0'
|
||||||
/*api 'cc.winboll.studio:winboll-shared:1.8.0'
|
implementation 'cc.winboll.studio:libappbase:15.11.0'
|
||||||
api 'io.github.medyo:android-about-page:2.0.0'
|
|
||||||
api 'com.jcraft:jsch:0.1.55'
|
//api fileTree(dir: 'libs', include: ['*.aar'])
|
||||||
api 'org.jsoup:jsoup:1.13.1'
|
api fileTree(dir: 'libs', include: ['*.jar'])
|
||||||
api 'com.squareup.okhttp3:okhttp:4.4.1'
|
|
||||||
|
|
||||||
api 'androidx.appcompat:appcompat:1.0.0'
|
|
||||||
api 'androidx.fragment:fragment:1.0.0'
|
|
||||||
api 'com.google.android.material:material:1.0.0'
|
|
||||||
|
|
||||||
api 'com.baoyz.pullrefreshlayout:library:1.2.0'
|
|
||||||
api 'com.github.getActivity:ToastUtils:10.5'
|
|
||||||
api 'io.github.medyo:android-about-page:2.0.0'
|
|
||||||
api 'org.jsoup:jsoup:1.13.1'
|
|
||||||
api 'com.squareup.okhttp3:okhttp:4.4.1'
|
|
||||||
|
|
||||||
api 'cc.winboll.studio:libaes:7.6.0'
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
#Created by .winboll/winboll_app_build.gradle
|
#Created by .winboll/winboll_app_build.gradle
|
||||||
#Wed Sep 03 20:59:53 HKT 2025
|
#Sun Nov 16 13:12:25 HKT 2025
|
||||||
stageCount=13
|
stageCount=1
|
||||||
libraryProject=
|
libraryProject=
|
||||||
baseVersion=15.4
|
baseVersion=15.11
|
||||||
publishVersion=15.4.12
|
publishVersion=15.11.0
|
||||||
buildCount=0
|
buildCount=0
|
||||||
baseBetaVersion=15.4.13
|
baseBetaVersion=15.11.1
|
||||||
|
|||||||
3
powerbell/proguard-rules.pro
vendored
3
powerbell/proguard-rules.pro
vendored
@@ -15,3 +15,6 @@
|
|||||||
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
|
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
|
||||||
# public *;
|
# public *;
|
||||||
#}
|
#}
|
||||||
|
|
||||||
|
## 米盟
|
||||||
|
-keep class com.miui.zeus.** { *; }
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<resources>
|
<resources>
|
||||||
|
|
||||||
<string name="app_name">能源钟☆</string>
|
<string name="app_name">PowerBell☆</string>
|
||||||
|
|
||||||
</resources>
|
</resources>
|
||||||
|
|||||||
@@ -1,7 +1,27 @@
|
|||||||
<?xml version='1.0' encoding='utf-8'?>
|
<?xml version='1.0' encoding='utf-8'?>
|
||||||
<manifest
|
<manifest
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
package="cc.winboll.studio.powerbell">
|
package="cc.winboll.studio.powerbell">
|
||||||
|
|
||||||
|
//米盟必要权限配置
|
||||||
|
<uses-permission android:name="android.permission.INTERNET"/>
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
|
||||||
|
<!-- 权限会用在部分下载类广告安装应用时使用 -->
|
||||||
|
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>
|
||||||
|
|
||||||
|
//以下为非必要权限配置,可以不申请,当您使用的米盟SDK版本大于等于5.3.1时可以选择性添加如下权限
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
|
||||||
|
<!-- 米盟提供“获取地理位置权限”和“不给予地理位置权限,开发者传入地理位置参数”两种方式上报用户位置,两种方式均可不选,无论通过何种方式提供给米盟用户地理位置,均需向用户声明地理位置权限将应用于米盟广告投放,米盟不强制获取地理位置信息-->
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
||||||
|
<!-- 米盟将通过此权限在Android 11系统上判定广告对应的应用是否在用户的app上安装,避免投放错误的广告,以此提高用户的广告体验。若添加此权限,需要在您的用户隐私文档中声明! -->
|
||||||
|
<uses-permission android:name="android.permission.QUERY_ALL_PACKAGES"/>
|
||||||
|
|
||||||
|
<!-- 通过GPS得到精确位置 -->
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
|
||||||
|
<!-- 通过网络得到粗略位置 -->
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
||||||
|
|
||||||
<!-- 拍摄照片和视频 -->
|
<!-- 拍摄照片和视频 -->
|
||||||
<uses-permission android:name="android.permission.CAMERA"/>
|
<uses-permission android:name="android.permission.CAMERA"/>
|
||||||
@@ -24,10 +44,29 @@
|
|||||||
<!-- 显示通知 -->
|
<!-- 显示通知 -->
|
||||||
<uses-permission android:name="android.permission.POST_NOTIFICATIONS"/>
|
<uses-permission android:name="android.permission.POST_NOTIFICATIONS"/>
|
||||||
|
|
||||||
|
<!-- PACKAGE_USAGE_STATS -->
|
||||||
|
<uses-permission android:name="android.permission.PACKAGE_USAGE_STATS"/>
|
||||||
|
|
||||||
|
<!-- BATTERY_STATS -->
|
||||||
|
<uses-permission android:name="android.permission.BATTERY_STATS"/>
|
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.POST_NOTIFICATIONS"/>
|
||||||
|
|
||||||
<uses-feature android:name="android.hardware.camera"/>
|
<uses-feature android:name="android.hardware.camera"/>
|
||||||
|
|
||||||
<uses-feature android:name="android.hardware.camera.autofocus"/>
|
<uses-feature android:name="android.hardware.camera.autofocus"/>
|
||||||
|
|
||||||
|
<!-- 1. 基础应用信息读取权限(Android 11 及以下) -->
|
||||||
|
<uses-permission android:name="android.permission.GET_PACKAGE_SIZE" />
|
||||||
|
|
||||||
|
<!-- 2. Android 11+ 应用列表读取权限(必须声明,否则无法获取全部应用) -->
|
||||||
|
<uses-permission android:name="android.permission.QUERY_ALL_PACKAGES"
|
||||||
|
tools:ignore="QueryAllPackagesPermission" />
|
||||||
|
|
||||||
|
<!-- 3. 可选:若需读取系统应用,添加此权限(部分机型需要) -->
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_PACKAGE_USAGE_STATS"
|
||||||
|
tools:ignore="ProtectedPermissions" />
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:name=".App"
|
android:name=".App"
|
||||||
android:allowBackup="true"
|
android:allowBackup="true"
|
||||||
@@ -36,7 +75,10 @@
|
|||||||
android:theme="@style/AppTheme_Default"
|
android:theme="@style/AppTheme_Default"
|
||||||
android:persistent="true"
|
android:persistent="true"
|
||||||
android:resizeableActivity="true"
|
android:resizeableActivity="true"
|
||||||
android:requestLegacyExternalStorage="true">
|
android:requestLegacyExternalStorage="true"
|
||||||
|
android:usesCleartextTraffic="true"
|
||||||
|
tools:ignore="GoogleAppIndexingWarning"
|
||||||
|
android:networkSecurityConfig="@xml/network_security_config">
|
||||||
|
|
||||||
<activity
|
<activity
|
||||||
android:name=".MainActivity"
|
android:name=".MainActivity"
|
||||||
@@ -123,6 +165,8 @@
|
|||||||
|
|
||||||
<activity android:name="cc.winboll.studio.powerbell.activities.PixelPickerActivity"/>
|
<activity android:name="cc.winboll.studio.powerbell.activities.PixelPickerActivity"/>
|
||||||
|
|
||||||
|
<activity android:name="cc.winboll.studio.powerbell.activities.BatteryReportActivity"/>
|
||||||
|
|
||||||
</application>
|
</application>
|
||||||
|
|
||||||
</manifest>
|
</manifest>
|
||||||
|
|||||||
@@ -4,10 +4,10 @@ import android.content.Context;
|
|||||||
import android.os.Environment;
|
import android.os.Environment;
|
||||||
import android.view.Gravity;
|
import android.view.Gravity;
|
||||||
import cc.winboll.studio.libappbase.GlobalApplication;
|
import cc.winboll.studio.libappbase.GlobalApplication;
|
||||||
|
import cc.winboll.studio.libappbase.ToastUtils;
|
||||||
import cc.winboll.studio.powerbell.receivers.GlobalApplicationReceiver;
|
import cc.winboll.studio.powerbell.receivers.GlobalApplicationReceiver;
|
||||||
import cc.winboll.studio.powerbell.utils.AppCacheUtils;
|
import cc.winboll.studio.powerbell.utils.AppCacheUtils;
|
||||||
import cc.winboll.studio.powerbell.utils.AppConfigUtils;
|
import cc.winboll.studio.powerbell.utils.AppConfigUtils;
|
||||||
import com.hjq.toast.ToastUtils;
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
|
||||||
public class App extends GlobalApplication {
|
public class App extends GlobalApplication {
|
||||||
@@ -50,7 +50,7 @@ public class App extends GlobalApplication {
|
|||||||
// 设置 Toast 布局样式
|
// 设置 Toast 布局样式
|
||||||
//ToastUtils.setView(R.layout.toast_custom_view);
|
//ToastUtils.setView(R.layout.toast_custom_view);
|
||||||
//ToastUtils.setStyle(new WhiteToastStyle());
|
//ToastUtils.setStyle(new WhiteToastStyle());
|
||||||
ToastUtils.setGravity(Gravity.BOTTOM, 0, 200);
|
//ToastUtils.setGravity(Gravity.BOTTOM, 0, 200);
|
||||||
|
|
||||||
// 设置数据配置存储工具
|
// 设置数据配置存储工具
|
||||||
_mAppConfigUtils = getAppConfigUtils(this);
|
_mAppConfigUtils = getAppConfigUtils(this);
|
||||||
@@ -77,5 +77,13 @@ public class App extends GlobalApplication {
|
|||||||
public void clearBatteryHistory() {
|
public void clearBatteryHistory() {
|
||||||
_mAppCacheUtils.clearBatteryHistory();
|
_mAppCacheUtils.clearBatteryHistory();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onTerminate() {
|
||||||
|
super.onTerminate();
|
||||||
|
ToastUtils.release();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,30 +3,69 @@ package cc.winboll.studio.powerbell;
|
|||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.app.Fragment;
|
import android.app.Fragment;
|
||||||
import android.app.FragmentTransaction;
|
import android.app.FragmentTransaction;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.DialogInterface;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.content.SharedPreferences;
|
||||||
import android.database.Cursor;
|
import android.database.Cursor;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.provider.MediaStore;
|
import android.provider.MediaStore;
|
||||||
|
import android.text.TextUtils;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.view.Display;
|
||||||
|
import android.view.Gravity;
|
||||||
|
import android.view.KeyEvent;
|
||||||
import android.view.Menu;
|
import android.view.Menu;
|
||||||
import android.view.MenuItem;
|
import android.view.MenuItem;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.view.Window;
|
||||||
|
import android.view.WindowManager;
|
||||||
|
import android.widget.FrameLayout;
|
||||||
import android.widget.RelativeLayout;
|
import android.widget.RelativeLayout;
|
||||||
import android.widget.Toast;
|
import android.widget.Toast;
|
||||||
import cc.winboll.studio.libaes.views.AToolbar;
|
import androidx.appcompat.app.AlertDialog;
|
||||||
|
import androidx.appcompat.widget.Toolbar;
|
||||||
|
import cc.winboll.studio.libappbase.LogActivity;
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
import cc.winboll.studio.powerbell.MainActivity;
|
import cc.winboll.studio.powerbell.MainActivity;
|
||||||
import cc.winboll.studio.powerbell.activities.AboutActivity;
|
import cc.winboll.studio.powerbell.activities.AboutActivity;
|
||||||
import cc.winboll.studio.powerbell.activities.BackgroundPictureActivity;
|
import cc.winboll.studio.powerbell.activities.BackgroundPictureActivity;
|
||||||
import cc.winboll.studio.powerbell.activities.BatteryReporterActivity;
|
import cc.winboll.studio.powerbell.activities.BatteryReportActivity;
|
||||||
import cc.winboll.studio.powerbell.activities.ClearRecordActivity;
|
import cc.winboll.studio.powerbell.activities.ClearRecordActivity;
|
||||||
import cc.winboll.studio.powerbell.activities.WinBoLLActivity;
|
import cc.winboll.studio.powerbell.activities.WinBoLLActivity;
|
||||||
import cc.winboll.studio.powerbell.beans.BackgroundPictureBean;
|
import cc.winboll.studio.powerbell.beans.BackgroundPictureBean;
|
||||||
import cc.winboll.studio.powerbell.fragments.MainViewFragment;
|
import cc.winboll.studio.powerbell.fragments.MainViewFragment;
|
||||||
import cc.winboll.studio.powerbell.utils.BackgroundPictureUtils;
|
import cc.winboll.studio.powerbell.utils.BackgroundPictureUtils;
|
||||||
|
import cc.winboll.studio.powerbell.utils.MimoUtils;
|
||||||
|
import com.miui.zeus.mimo.sdk.ADParams;
|
||||||
|
import com.miui.zeus.mimo.sdk.BannerAd;
|
||||||
|
import com.miui.zeus.mimo.sdk.MimoCustomController;
|
||||||
|
import com.miui.zeus.mimo.sdk.MimoLocation;
|
||||||
|
import com.miui.zeus.mimo.sdk.MimoSdk;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
public class MainActivity extends WinBoLLActivity {
|
public class MainActivity extends WinBoLLActivity {
|
||||||
|
|
||||||
public static final String TAG = "MainActivity";
|
public static final String TAG = "MainActivity";
|
||||||
|
|
||||||
|
private static final String PRIVACY_FILE = "privacy_pfs";
|
||||||
|
private static final String PRIVACY_VALUE = "privacy_value";//0: 拒绝,1:赞同
|
||||||
|
|
||||||
|
private SharedPreferences mSharedPreferences;
|
||||||
|
|
||||||
|
private String BANNER_POS_ID = "802e356f1726f9ff39c69308bfd6f06a";
|
||||||
|
private String BANNER_POS_ID_WINBOLL = "d129ee5a263911f981a6dc7a9802e3e7";
|
||||||
|
private BannerAd mBannerAd;
|
||||||
|
private List<BannerAd> mAllBanners = new ArrayList<>();
|
||||||
|
|
||||||
|
private ViewGroup mContainer;
|
||||||
|
|
||||||
|
|
||||||
|
private boolean mIsBiddingWin = true;
|
||||||
|
|
||||||
public static final int BACKGROUND_PICTURE_REQUEST_CODE = 0;
|
public static final int BACKGROUND_PICTURE_REQUEST_CODE = 0;
|
||||||
|
|
||||||
public static MainActivity _mMainActivity;
|
public static MainActivity _mMainActivity;
|
||||||
@@ -37,7 +76,7 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
Menu mMenu;
|
Menu mMenu;
|
||||||
Fragment mCurrentShowFragment;
|
Fragment mCurrentShowFragment;
|
||||||
MainViewFragment mMainViewFragment;
|
MainViewFragment mMainViewFragment;
|
||||||
AToolbar mAToolbar;
|
Toolbar mToolbar;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Activity getActivity() {
|
public Activity getActivity() {
|
||||||
@@ -51,9 +90,14 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
//LogUtils.d(TAG, "onCreate(...)");
|
LogUtils.d(TAG, "onCreate(...)");
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
setContentView(R.layout.activity_main);
|
setContentView(R.layout.activity_main);
|
||||||
|
mContainer = findViewById(R.id.ads_container);
|
||||||
|
|
||||||
|
// 米盟模块
|
||||||
|
showPrivacy();
|
||||||
|
|
||||||
|
|
||||||
// 设置调试日志
|
// 设置调试日志
|
||||||
// mLogView = findViewById(R.id.logview);
|
// mLogView = findViewById(R.id.logview);
|
||||||
@@ -66,10 +110,10 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
//mAppConfigUtils = AppConfigUtils.getInstance(mApplication);
|
//mAppConfigUtils = AppConfigUtils.getInstance(mApplication);
|
||||||
|
|
||||||
// 初始化工具栏
|
// 初始化工具栏
|
||||||
mAToolbar = (AToolbar) findViewById(R.id.toolbar);
|
mToolbar = findViewById(R.id.toolbar);
|
||||||
setActionBar(mAToolbar);
|
setSupportActionBar(mToolbar);
|
||||||
//mAToolbar.setSubtitle("Main");
|
//mAToolbar.setSubtitle("Main");
|
||||||
mAToolbar.setTitleTextAppearance(this, R.style.Toolbar_TitleText);
|
mToolbar.setTitleTextAppearance(this, R.style.Toolbar_TitleText);
|
||||||
|
|
||||||
if (mMainViewFragment == null) {
|
if (mMainViewFragment == null) {
|
||||||
FragmentTransaction tx = getFragmentManager().beginTransaction();
|
FragmentTransaction tx = getFragmentManager().beginTransaction();
|
||||||
@@ -82,9 +126,143 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
// NotificationHelper notificationUtils = new NotificationHelper(this);
|
// NotificationHelper notificationUtils = new NotificationHelper(this);
|
||||||
// notificationUtils.createNotificationChannels();
|
// notificationUtils.createNotificationChannels();
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// public void onADs(View view) {
|
||||||
|
// LogUtils.d(TAG, "onPostCreate");
|
||||||
|
// fetchAd();
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*public void showADs() {
|
||||||
|
//(5.3.4新增接口) 请使用最新接口集成
|
||||||
|
ADParams params = new ADParams.Builder().setUpId(BANNER_POS_ID).build();
|
||||||
|
bannerAd.loadAd(params, new BannerAd.BannerLoadListener() {
|
||||||
|
//请求成功回调
|
||||||
|
@Override
|
||||||
|
public void onBannerAdLoadSuccess() {
|
||||||
|
}
|
||||||
|
//请求失败回调
|
||||||
|
@Override
|
||||||
|
public void onAdLoadFailed (int errorCode, String errorMsg) {
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
////废弃 将会在后面版本删除
|
||||||
|
// @Deprecated
|
||||||
|
// bannerAd.loadAd(upId, new BannerAd.BannerLoadListener() {
|
||||||
|
// //请求成功回调
|
||||||
|
// @Override
|
||||||
|
// public void onBannerAdLoadSuccess() {
|
||||||
|
// }
|
||||||
|
// //请求失败回调
|
||||||
|
// @Override
|
||||||
|
// public void onAdLoadFailed (int errorCode, String errorMsg) {
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
|
||||||
|
bannerAd.setDownLoadListener(new BannerAd.BannerDownloadListener() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadStarted() {
|
||||||
|
//开始下载
|
||||||
|
LogUtils.d(TAG, "ads 开始下载");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadProgressUpdated(int progress) {
|
||||||
|
//下载进度,例如:${progress}%
|
||||||
|
LogUtils.d(TAG, String.format("ads 下载进度 %d", progress));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadPaused() {
|
||||||
|
//下载暂停
|
||||||
|
LogUtils.d(TAG, "ads 下载暂停");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadCancel() {
|
||||||
|
//取消下载
|
||||||
|
LogUtils.d(TAG, "ads 取消下载");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadFailed(int errorCode) {
|
||||||
|
//下载失败, 若需要了解errorCode具体含义,请咨询米盟
|
||||||
|
LogUtils.d(TAG, String.format("ads 下载失败。errorCode :%d", errorCode));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadFinished() {
|
||||||
|
//下载结束
|
||||||
|
LogUtils.d(TAG, "ads 下载结束");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInstallStart() {
|
||||||
|
//开始安装
|
||||||
|
LogUtils.d(TAG, "ads 开始安装");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInstallFailed(int errorCode) {
|
||||||
|
//安装失败
|
||||||
|
LogUtils.d(TAG, "ads 安装失败");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInstallSuccess() {
|
||||||
|
//安装成功
|
||||||
|
LogUtils.d(TAG, "ads 安装成功");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
LinearLayout ads_container = findViewById(R.id.ads_container);
|
||||||
|
bannerAd.showAd(this, ads_container, new BannerAd.BannerInteractionListener() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdClick() {
|
||||||
|
// 广告被点击
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdShow() {
|
||||||
|
// 广告被展示
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdDismiss() {
|
||||||
|
// 点击关闭按钮广告消失回调
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRenderSuccess() {
|
||||||
|
// 广告渲染成功
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRenderFail(int code, String msg) {
|
||||||
|
// 广告渲染失败
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
}*/
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onDestroy() {
|
||||||
|
super.onDestroy();
|
||||||
|
//bannerAd.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void showFragment(Fragment fragment) {
|
void showFragment(Fragment fragment) {
|
||||||
FragmentTransaction tx = getFragmentManager().beginTransaction();
|
FragmentTransaction tx = getFragmentManager().beginTransaction();
|
||||||
for (Fragment item : getFragmentManager().getFragments()) {
|
for (Fragment item : getFragmentManager().getFragments()) {
|
||||||
@@ -133,6 +311,24 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
//NotificationHelper.cancelRemindNotification(this);
|
//NotificationHelper.cancelRemindNotification(this);
|
||||||
reloadBackground();
|
reloadBackground();
|
||||||
setBackgroundColor();
|
setBackgroundColor();
|
||||||
|
|
||||||
|
//LogUtils.d(TAG, "ads start");
|
||||||
|
new Thread(new Runnable(){
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
//LogUtils.d(TAG, "run() 1");
|
||||||
|
runOnUiThread(new Runnable(){
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
//LogUtils.d(TAG, "run() 2");
|
||||||
|
//try {
|
||||||
|
//Thread.sleep(1000);
|
||||||
|
fetchAd();
|
||||||
|
//} catch (InterruptedException e) {}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Menu icons are inflated just as they were with actionbar
|
// Menu icons are inflated just as they were with actionbar
|
||||||
@@ -159,9 +355,9 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
if (menuItemId == R.id.action_about) {
|
if (menuItemId == R.id.action_about) {
|
||||||
Intent intent = new Intent(this, AboutActivity.class);
|
Intent intent = new Intent(this, AboutActivity.class);
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
} else if (menuItemId == R.id.action_battery_reporter) {
|
} else if (menuItemId == R.id.action_battery_report) {
|
||||||
Intent intent = new Intent();
|
Intent intent = new Intent();
|
||||||
intent.setClass(this, BatteryReporterActivity.class);
|
intent.setClass(this, BatteryReportActivity.class);
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
} else if (menuItemId == R.id.action_clearrecord) {
|
} else if (menuItemId == R.id.action_clearrecord) {
|
||||||
Intent intent = new Intent();
|
Intent intent = new Intent();
|
||||||
@@ -172,7 +368,7 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
intent.setClass(this, BackgroundPictureActivity.class);
|
intent.setClass(this, BackgroundPictureActivity.class);
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
} else if (menuItemId == R.id.action_log) {
|
} else if (menuItemId == R.id.action_log) {
|
||||||
App.getWinBoLLActivityManager().startLogActivity(this);
|
LogActivity.startLogActivity(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@@ -217,4 +413,275 @@ public class MainActivity extends WinBoLLActivity {
|
|||||||
RelativeLayout mainLayout = findViewById(R.id.activitymainRelativeLayout1);
|
RelativeLayout mainLayout = findViewById(R.id.activitymainRelativeLayout1);
|
||||||
mainLayout.setBackgroundColor(nPixelColor);
|
mainLayout.setBackgroundColor(nPixelColor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void showAd() {
|
||||||
|
LogUtils.d(TAG, "showAd()");
|
||||||
|
final FrameLayout container = new FrameLayout(this);
|
||||||
|
container.setPadding(0, 0, 0, MimoUtils.dpToPx(this, 10));
|
||||||
|
mContainer.addView(container, new FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.WRAP_CONTENT));
|
||||||
|
|
||||||
|
if (mIsBiddingWin) {
|
||||||
|
//如果竞价成功,传⼊本次实际结算价
|
||||||
|
mBannerAd.setPrice(getPrice());
|
||||||
|
}
|
||||||
|
mBannerAd.showAd(this, container, new BannerAd.BannerInteractionListener() {
|
||||||
|
@Override
|
||||||
|
public void onAdClick() {
|
||||||
|
LogUtils.d(TAG, "onAdClick");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdShow() {
|
||||||
|
//mShowBtn.setEnabled(false);
|
||||||
|
LogUtils.d(TAG, "onAdShow");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdDismiss() {
|
||||||
|
LogUtils.d(TAG, "onAdDismiss");
|
||||||
|
mContainer.removeView(container);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRenderSuccess() {
|
||||||
|
LogUtils.d(TAG, "onRenderSuccess");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRenderFail(int code, String msg) {
|
||||||
|
LogUtils.e(TAG, "onRenderFail errorCode " + code + " errorMsg " + msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void fetchAd() {
|
||||||
|
LogUtils.d(TAG, "fetchAd()");
|
||||||
|
mBannerAd = new BannerAd();
|
||||||
|
mAllBanners.add(mBannerAd);
|
||||||
|
mBannerAd.setDownLoadListener(new BannerAd.BannerDownloadListener() {
|
||||||
|
@Override
|
||||||
|
public void onDownloadStarted() {
|
||||||
|
LogUtils.d(TAG, "onDownloadStarted");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadPaused() {
|
||||||
|
LogUtils.d(TAG, "onDownloadPaused");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadFailed(int errorCode) {
|
||||||
|
LogUtils.d(TAG, "onDownloadFailed, errorCode = " + errorCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadFinished() {
|
||||||
|
LogUtils.d(TAG, "onDownloadFinished");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadProgressUpdated(int progress) {
|
||||||
|
LogUtils.d(TAG, "onDownloadProgressUpdated " + progress + "%");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInstallFailed(int errorCode) {
|
||||||
|
LogUtils.d(TAG, "onInstallFailed, errorCode = " + errorCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInstallStart() {
|
||||||
|
LogUtils.d(TAG, "onInstallStart");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInstallSuccess() {
|
||||||
|
LogUtils.d(TAG, "onInstallSuccess");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDownloadCancel() {
|
||||||
|
LogUtils.d(TAG, "onDownloadCancel");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
String currentAD_ID = isEvenSecond()?BANNER_POS_ID_WINBOLL: BANNER_POS_ID;
|
||||||
|
LogUtils.d(TAG, String.format("currentAD_ID = %s", currentAD_ID));
|
||||||
|
ADParams params = new ADParams.Builder().setUpId(currentAD_ID).build();
|
||||||
|
mBannerAd.loadAd(params, new BannerAd.BannerLoadListener() {
|
||||||
|
@Override
|
||||||
|
public void onBannerAdLoadSuccess() {
|
||||||
|
LogUtils.d(TAG, "onBannerAdLoadSuccess()");
|
||||||
|
showAd();
|
||||||
|
//mShowBtn.setEnabled(true);
|
||||||
|
|
||||||
|
/*long bannerPrice = getPrice();
|
||||||
|
if (mIsBiddingWin) {
|
||||||
|
//竞价成功时候上报win
|
||||||
|
Map<String, Long> auctionBidInfo = new HashMap<>();
|
||||||
|
auctionBidInfo.put(BaseAd.IBidding.EXPECT_COST_PRICE, bannerPrice);
|
||||||
|
auctionBidInfo.put(BaseAd.IBidding.HIGHEST_LOSS_PRICE, BID_WIN_HIGHEST_LOSS_PRICE);
|
||||||
|
mBannerAd.win(auctionBidInfo);
|
||||||
|
} else {
|
||||||
|
//竞价失败的时候上报loss
|
||||||
|
Map<String, Object> lossReasonInfo = new HashMap<>();
|
||||||
|
lossReasonInfo.put(BaseAd.IBidding.WIN_PRICE, BID_LOSS_WIN_PRICE);
|
||||||
|
lossReasonInfo.put(BaseAd.IBidding.LOSS_REASON, BaseAd.LossReason.TYPE_LOWER_OTHER_BIDDER_PRICE);
|
||||||
|
lossReasonInfo.put(BaseAd.IBidding.ADN_ID, 1);
|
||||||
|
mBannerAd.loss(lossReasonInfo);
|
||||||
|
}*/
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdLoadFailed(int errorCode, String errorMsg) {
|
||||||
|
LogUtils.d(TAG, "errorCode = " + errorCode + ", errorMsg = " + errorMsg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 判断当前时间是单数秒还是双数秒
|
||||||
|
* @return true:双数秒(0、2、4...),false:单数秒(1、3、5...)
|
||||||
|
*/
|
||||||
|
private boolean isEvenSecond() {
|
||||||
|
// 1. 获取当前毫秒级时间戳
|
||||||
|
long currentTimeMillis = System.currentTimeMillis();
|
||||||
|
// 2. 转换为秒数(毫秒 ÷ 1000,向下取整)
|
||||||
|
long currentSecond = currentTimeMillis / 1000;
|
||||||
|
// 3. 对 2 取余:余数为 0 是双数秒,余数为 1 是单数秒
|
||||||
|
return currentSecond % 2 == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 扩展:获取当前秒数的奇偶描述(直接返回文字说明)
|
||||||
|
* @return 字符串:"单数秒" 或 "双数秒"
|
||||||
|
*/
|
||||||
|
// private String getSecondParityDesc() {
|
||||||
|
// long currentSecond = System.currentTimeMillis() / 1000;
|
||||||
|
// return (currentSecond % 2 == 0) ? "双数秒" : "单数秒";
|
||||||
|
// }
|
||||||
|
|
||||||
|
|
||||||
|
private long getPrice() {
|
||||||
|
if (mBannerAd == null) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
Map<String, Object> map = mBannerAd.getMediaExtraInfo();
|
||||||
|
if (map == null || map.isEmpty()) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return (long) map.get("price");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void showPrivacy() {
|
||||||
|
String privacyAgreeValue = getSharedPreferences().getString(PRIVACY_VALUE, null);
|
||||||
|
if (TextUtils.equals(privacyAgreeValue, String.valueOf(0))) {
|
||||||
|
Log.i(TAG, "已拒绝隐私协议,广告已处于不可以状态...");
|
||||||
|
Toast.makeText(this, "已拒绝隐私协议,广告已处于不可以状态", Toast.LENGTH_SHORT).show();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (TextUtils.equals(privacyAgreeValue, String.valueOf(1))) {
|
||||||
|
Log.i(TAG, "已同意隐私协议,开始初始化...");
|
||||||
|
initMimoSdk();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Log.i(TAG, "开始弹出隐私协议...");
|
||||||
|
AlertDialog.Builder builder = new AlertDialog.Builder(MainActivity.this);
|
||||||
|
builder.setTitle("用户须知");
|
||||||
|
builder.setMessage("小米广告SDK隐私政策: https://dev.mi.com/distribute/doc/details?pId=1688, 请复制到浏览器查看");
|
||||||
|
builder.setIcon(R.drawable.ic_launcher);
|
||||||
|
builder.setCancelable(false); //点击对话框以外的区域是否让对话框消失
|
||||||
|
builder.setPositiveButton("同意", new DialogInterface.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(DialogInterface dialog, int which) {
|
||||||
|
SharedPreferences.Editor editor = getSharedPreferences().edit();
|
||||||
|
editor.putString(PRIVACY_VALUE, String.valueOf(1));
|
||||||
|
editor.apply();
|
||||||
|
initMimoSdk();
|
||||||
|
dialog.dismiss();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
builder.setNegativeButton("拒绝", new DialogInterface.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(DialogInterface dialog, int which) {
|
||||||
|
SharedPreferences.Editor editor = getSharedPreferences().edit();
|
||||||
|
editor.putString(PRIVACY_VALUE, String.valueOf(0));
|
||||||
|
editor.apply();
|
||||||
|
dialog.dismiss();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
AlertDialog dialog = builder.create();
|
||||||
|
|
||||||
|
Window window = dialog.getWindow();
|
||||||
|
window.setGravity(Gravity.BOTTOM);
|
||||||
|
|
||||||
|
WindowManager m = getWindowManager();
|
||||||
|
Display d = m.getDefaultDisplay();
|
||||||
|
WindowManager.LayoutParams p = dialog.getWindow().getAttributes();
|
||||||
|
p.width = d.getWidth();
|
||||||
|
dialog.getWindow().setAttributes(p);
|
||||||
|
dialog.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void initMimoSdk() {
|
||||||
|
MimoSdk.init(this, new MimoCustomController() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isCanUseLocation() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MimoLocation getMimoLocation() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isCanUseWifiState() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean alist() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} , new MimoSdk.InitCallback() {
|
||||||
|
@Override
|
||||||
|
public void success() {
|
||||||
|
Log.d(TAG, "MimoSdk init success");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void fail(int code, String msg) {
|
||||||
|
Log.e(TAG, "MimoSdk init fail, code=" + code + ",msg=" + msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
MimoSdk.setDebugOn(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean dispatchKeyEvent(KeyEvent event) {
|
||||||
|
return super.dispatchKeyEvent(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
// @Override
|
||||||
|
// public boolean onNavigationItemSelected(@NonNull MenuItem menuItem) {
|
||||||
|
// return false;
|
||||||
|
// }
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setupToolbar() {
|
||||||
|
super.setupToolbar();
|
||||||
|
if (getSupportActionBar() != null) {
|
||||||
|
getSupportActionBar().setDisplayHomeAsUpEnabled(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public SharedPreferences getSharedPreferences() {
|
||||||
|
if (mSharedPreferences == null) {
|
||||||
|
mSharedPreferences = getSharedPreferences(PRIVACY_FILE, Context.MODE_PRIVATE);
|
||||||
|
}
|
||||||
|
return mSharedPreferences;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,9 +10,9 @@ import android.content.Context;
|
|||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
import android.widget.LinearLayout;
|
import android.widget.LinearLayout;
|
||||||
|
import cc.winboll.studio.libaes.models.APPInfo;
|
||||||
import cc.winboll.studio.libaes.views.AToolbar;
|
import cc.winboll.studio.libaes.views.AToolbar;
|
||||||
import cc.winboll.studio.libaes.winboll.APPInfo;
|
import cc.winboll.studio.libaes.views.AboutView;
|
||||||
import cc.winboll.studio.libaes.winboll.AboutView;
|
|
||||||
import cc.winboll.studio.powerbell.R;
|
import cc.winboll.studio.powerbell.R;
|
||||||
|
|
||||||
public class AboutActivity extends Activity {
|
public class AboutActivity extends Activity {
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ import androidx.core.app.ActivityCompat;
|
|||||||
import androidx.core.content.ContextCompat;
|
import androidx.core.content.ContextCompat;
|
||||||
import cc.winboll.studio.libaes.views.AToolbar;
|
import cc.winboll.studio.libaes.views.AToolbar;
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
import cc.winboll.studio.libappbase.utils.ToastUtils;
|
import cc.winboll.studio.libappbase.ToastUtils;
|
||||||
import cc.winboll.studio.powerbell.App;
|
import cc.winboll.studio.powerbell.App;
|
||||||
import cc.winboll.studio.powerbell.R;
|
import cc.winboll.studio.powerbell.R;
|
||||||
import cc.winboll.studio.powerbell.beans.BackgroundPictureBean;
|
import cc.winboll.studio.powerbell.beans.BackgroundPictureBean;
|
||||||
|
|||||||
@@ -0,0 +1,516 @@
|
|||||||
|
package cc.winboll.studio.powerbell.activities;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @Author ZhanGSKen&豆包大模型<zhangsken@qq.com>
|
||||||
|
* @Date 2025/10/22 13:21
|
||||||
|
* @Describe BatteryReportActivity
|
||||||
|
*/
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.BroadcastReceiver;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.IntentFilter;
|
||||||
|
import android.content.pm.ApplicationInfo;
|
||||||
|
import android.content.pm.PackageManager;
|
||||||
|
import android.os.Build;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.provider.Settings;
|
||||||
|
import android.text.Editable;
|
||||||
|
import android.text.TextWatcher;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.EditText;
|
||||||
|
import android.widget.TextView;
|
||||||
|
import android.widget.Toast;
|
||||||
|
import androidx.recyclerview.widget.LinearLayoutManager;
|
||||||
|
import androidx.recyclerview.widget.RecyclerView;
|
||||||
|
import cc.winboll.studio.powerbell.R;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
|
|
||||||
|
public class BatteryReportActivity extends Activity {
|
||||||
|
public static final String TAG = "BatteryReportActivity";
|
||||||
|
|
||||||
|
private RecyclerView rvBatteryReport;
|
||||||
|
private BatteryReportAdapter adapter;
|
||||||
|
private List<AppBatteryModel> dataList = new ArrayList<AppBatteryModel>();
|
||||||
|
private List<AppBatteryModel> filteredList = new ArrayList<AppBatteryModel>();
|
||||||
|
private BroadcastReceiver batteryReceiver;
|
||||||
|
private int batteryCapacity = 5400; // 电池容量(mAh)
|
||||||
|
private float lastBatteryPercent = 100.0f;
|
||||||
|
private long lastCheckTime = System.currentTimeMillis();
|
||||||
|
private EditText etSearch;
|
||||||
|
private Map<String, Long> appRunTimeCache = new HashMap<String, Long>();
|
||||||
|
private Map<String, String> packageToAppNameCache = new HashMap<String, String>();
|
||||||
|
private PackageManager mPackageManager;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
setContentView(R.layout.activity_battery_report);
|
||||||
|
mPackageManager = getPackageManager();
|
||||||
|
|
||||||
|
// 权限检查(Java7 传统条件判断)
|
||||||
|
if (!hasUsageStatsPermission(this)) {
|
||||||
|
Toast.makeText(this, "请进入设置-应用-权限-特殊访问权限-使用情况访问权限,开启本应用的权限", Toast.LENGTH_LONG).show();
|
||||||
|
startActivity(new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
etSearch = (EditText) findViewById(R.id.et_search);
|
||||||
|
rvBatteryReport = (RecyclerView) findViewById(R.id.rv_battery_report);
|
||||||
|
rvBatteryReport.setLayoutManager(new LinearLayoutManager(this));
|
||||||
|
|
||||||
|
// 初始化流程:新增“加载24小时累计耗电”步骤
|
||||||
|
loadAllAppPackage();
|
||||||
|
preCacheAllAppNames();
|
||||||
|
appRunTimeCache = getAppRunTime();
|
||||||
|
updateAppRunTimeToModel();
|
||||||
|
calculateInitial24hTotalConsumption(); // 初始化时计算24小时累计耗电
|
||||||
|
filteredList.addAll(dataList);
|
||||||
|
adapter = new BatteryReportAdapter(this, filteredList, mPackageManager, packageToAppNameCache);
|
||||||
|
rvBatteryReport.setAdapter(adapter);
|
||||||
|
|
||||||
|
// 搜索监听(不变)
|
||||||
|
etSearch.addTextChangedListener(new TextWatcher() {
|
||||||
|
@Override
|
||||||
|
public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onTextChanged(CharSequence s, int start, int before, int count) {
|
||||||
|
filterAppsByPackageAndName(s.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void afterTextChanged(Editable s) {}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 电池广播:调用修改后的“单次耗电计算+累计累加”方法
|
||||||
|
batteryReceiver = new BroadcastReceiver() {
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context context, Intent intent) {
|
||||||
|
int level = intent.getIntExtra("level", 100);
|
||||||
|
int scale = intent.getIntExtra("scale", 100);
|
||||||
|
float currentPercent = (float) level / scale * 100;
|
||||||
|
LogUtils.d(TAG, "电池百分比变化:" + lastBatteryPercent + " -> " + currentPercent);
|
||||||
|
|
||||||
|
if (currentPercent < lastBatteryPercent) {
|
||||||
|
float dropPercent = lastBatteryPercent - currentPercent;
|
||||||
|
long duration = System.currentTimeMillis() - lastCheckTime;
|
||||||
|
LogUtils.d(TAG, "电池消耗:" + dropPercent + "%,时长:" + duration + "ms");
|
||||||
|
appRunTimeCache = getAppRunTime();
|
||||||
|
updateAppRunTimeToModel();
|
||||||
|
calculateSingleConsumptionAndAccumulate(dropPercent, appRunTimeCache); // 单次+累计逻辑
|
||||||
|
}
|
||||||
|
|
||||||
|
lastBatteryPercent = currentPercent;
|
||||||
|
lastCheckTime = System.currentTimeMillis();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
registerReceiver(batteryReceiver, new IntentFilter(Intent.ACTION_BATTERY_CHANGED));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onDestroy() {
|
||||||
|
super.onDestroy();
|
||||||
|
// Java7 显式非空判断
|
||||||
|
if (batteryReceiver != null) {
|
||||||
|
unregisterReceiver(batteryReceiver);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 加载所有应用(仅获取包名,初始化模型时单次耗电、累计耗电均设为0)
|
||||||
|
*/
|
||||||
|
private void loadAllAppPackage() {
|
||||||
|
List<ApplicationInfo> appList = mPackageManager.getInstalledApplications(PackageManager.GET_META_DATA);
|
||||||
|
dataList.clear();
|
||||||
|
|
||||||
|
LogUtils.d(TAG, "开始加载应用包名列表,共找到" + appList.size() + "个应用");
|
||||||
|
|
||||||
|
for (ApplicationInfo appInfo : appList) {
|
||||||
|
String packageName = appInfo.packageName;
|
||||||
|
// 初始化:单次耗电(consumption)=0,累计耗电(totalConsumption)=0,运行时长=0
|
||||||
|
dataList.add(new AppBatteryModel(packageName, 0.0f, 0.0f, 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
LogUtils.d(TAG, "应用包名列表加载完成,共添加" + dataList.size() + "个包名。");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 预缓存应用名称(逻辑不变)
|
||||||
|
*/
|
||||||
|
private void preCacheAllAppNames() {
|
||||||
|
packageToAppNameCache.clear();
|
||||||
|
LogUtils.d(TAG, "开始预缓存包名-应用名称映射");
|
||||||
|
|
||||||
|
for (AppBatteryModel model : dataList) {
|
||||||
|
String packageName = model.getPackageName();
|
||||||
|
String appName = getAppNameByPackage(packageName);
|
||||||
|
packageToAppNameCache.put(packageName, appName);
|
||||||
|
}
|
||||||
|
|
||||||
|
LogUtils.d(TAG, "预缓存完成,共缓存" + packageToAppNameCache.size() + "个应用名称");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 通过包名获取应用名称(逻辑不变)
|
||||||
|
*/
|
||||||
|
private String getAppNameByPackage(String packageName) {
|
||||||
|
try {
|
||||||
|
ApplicationInfo appInfo = mPackageManager.getApplicationInfo(packageName, 0);
|
||||||
|
return mPackageManager.getApplicationLabel(appInfo).toString();
|
||||||
|
} catch (PackageManager.NameNotFoundException e) {
|
||||||
|
LogUtils.e(TAG, "包名" + packageName + "对应的应用未找到:" + e.getMessage());
|
||||||
|
return packageName;
|
||||||
|
} catch (Exception e) {
|
||||||
|
LogUtils.e(TAG, "查询应用名称失败(包名:" + packageName + "):" + e.getMessage());
|
||||||
|
return packageName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 更新运行时长到模型(逻辑不变)
|
||||||
|
*/
|
||||||
|
private void updateAppRunTimeToModel() {
|
||||||
|
int nCount = 0;
|
||||||
|
for (AppBatteryModel model : dataList) {
|
||||||
|
String packageName = model.getPackageName();
|
||||||
|
Long runTime;
|
||||||
|
if (appRunTimeCache.containsKey(packageName)) {
|
||||||
|
runTime = appRunTimeCache.get(packageName);
|
||||||
|
LogUtils.d(TAG, String.format("应用包 %s 运行时长已更新。", packageName));
|
||||||
|
nCount++;
|
||||||
|
} else {
|
||||||
|
runTime = 0L;
|
||||||
|
}
|
||||||
|
model.setRunTime(runTime);
|
||||||
|
}
|
||||||
|
LogUtils.d(TAG, String.format("dataList.size() %d, appRunTimeCache.size() %d。", dataList.size(), appRunTimeCache.size()));
|
||||||
|
LogUtils.d(TAG, String.format("updateAppRunTimeToModel() 更新的数据量为:%d", nCount));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 【新增】初始化时计算24小时累计耗电(赋值给totalConsumption)
|
||||||
|
* 逻辑:基于24小时运行时长占比,分配当前电池容量的理论24小时消耗
|
||||||
|
*/
|
||||||
|
private void calculateInitial24hTotalConsumption() {
|
||||||
|
long total24hRunTime = 0;
|
||||||
|
// 1. 计算24小时内所有应用总运行时长
|
||||||
|
for (Map.Entry<String, Long> entry : appRunTimeCache.entrySet()) {
|
||||||
|
total24hRunTime += entry.getValue();
|
||||||
|
}
|
||||||
|
LogUtils.d(TAG, "24小时内所有应用总运行时长:" + formatRunTime(total24hRunTime));
|
||||||
|
|
||||||
|
// 2. 按运行时长占比分配24小时累计耗电(假设电池满电循环,用总容量近似24小时总消耗)
|
||||||
|
for (AppBatteryModel model : dataList) {
|
||||||
|
String packageName = model.getPackageName();
|
||||||
|
Long app24hRunTime = appRunTimeCache.getOrDefault(packageName, 0L);
|
||||||
|
|
||||||
|
// 计算占比与累计耗电
|
||||||
|
float ratio = (total24hRunTime > 0) ? (float) app24hRunTime / total24hRunTime : 0;
|
||||||
|
float initialTotalConsumption = batteryCapacity * ratio; // 用电池容量近似24小时总消耗
|
||||||
|
model.setTotalConsumption(initialTotalConsumption); // 初始化累计耗电
|
||||||
|
LogUtils.d(TAG, String.format("应用包 %s 24小时累计耗电初始化:%.1f mAh", packageName, initialTotalConsumption));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 【核心修改】计算单次耗电(赋值给consumption)+ 累加至累计耗电(totalConsumption = totalConsumption + consumption)
|
||||||
|
*/
|
||||||
|
private void calculateSingleConsumptionAndAccumulate(float dropPercent, Map<String, Long> runTimeMap) {
|
||||||
|
long totalSingleRunTime = 0;
|
||||||
|
// 1. 计算本次电池下降期间的总运行时长
|
||||||
|
for (Map.Entry<String, Long> entry : runTimeMap.entrySet()) {
|
||||||
|
totalSingleRunTime += entry.getValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. 遍历计算每个应用的“单次耗电”并“累加至累计”
|
||||||
|
for (AppBatteryModel model : dataList) {
|
||||||
|
String packageName = model.getPackageName();
|
||||||
|
Long appSingleRunTime = runTimeMap.getOrDefault(packageName, 0L);
|
||||||
|
|
||||||
|
// 步骤1:计算本次单次耗电(赋值给consumption)
|
||||||
|
float ratio = (totalSingleRunTime > 0) ? (float) appSingleRunTime / totalSingleRunTime : 0;
|
||||||
|
float singleConsumption = batteryCapacity * dropPercent / 100 * ratio; // 单次消耗
|
||||||
|
model.setConsumption(singleConsumption); // 存储单次耗电
|
||||||
|
|
||||||
|
// 步骤2:累加单次耗电到累计耗电(totalConsumption = 原有累计 + 本次单次)
|
||||||
|
float newTotalConsumption = model.getTotalConsumption() + singleConsumption;
|
||||||
|
model.setTotalConsumption(newTotalConsumption); // 更新累计耗电
|
||||||
|
|
||||||
|
// 同步运行时长
|
||||||
|
model.setRunTime(appSingleRunTime);
|
||||||
|
|
||||||
|
LogUtils.d(TAG, String.format("应用包 %s:单次耗电%.1f mAh,累计耗电%.1f mAh",
|
||||||
|
packageName, singleConsumption, newTotalConsumption));
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. 按累计耗电排序(从高到低)
|
||||||
|
Collections.sort(dataList, new Comparator<AppBatteryModel>() {
|
||||||
|
@Override
|
||||||
|
public int compare(AppBatteryModel m1, AppBatteryModel m2) {
|
||||||
|
return Float.compare(m2.getTotalConsumption(), m1.getTotalConsumption());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 4. 重新应用过滤并刷新列表
|
||||||
|
filterAppsByPackageAndName(etSearch.getText().toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 双维度过滤(逻辑不变)
|
||||||
|
*/
|
||||||
|
private void filterAppsByPackageAndName(String keyword) {
|
||||||
|
filteredList.clear();
|
||||||
|
if (keyword == null || keyword.isEmpty()) {
|
||||||
|
filteredList.addAll(dataList);
|
||||||
|
} else {
|
||||||
|
String lowerKeyword = keyword.toLowerCase();
|
||||||
|
|
||||||
|
for (AppBatteryModel model : dataList) {
|
||||||
|
String packageName = model.getPackageName();
|
||||||
|
String packageNameLower = packageName.toLowerCase();
|
||||||
|
String appName = packageToAppNameCache.get(packageName);
|
||||||
|
String appNameLower = appName.toLowerCase();
|
||||||
|
|
||||||
|
boolean isMatched = packageNameLower.contains(lowerKeyword)
|
||||||
|
|| appNameLower.contains(lowerKeyword);
|
||||||
|
|
||||||
|
if (isMatched) {
|
||||||
|
filteredList.add(model);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
adapter.notifyDataSetChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取应用运行时长(逻辑不变,返回24小时运行时长)
|
||||||
|
*/
|
||||||
|
private Map<String, Long> getAppRunTime() {
|
||||||
|
Map<String, Long> runTimeMap = new HashMap<String, Long>();
|
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
|
||||||
|
try {
|
||||||
|
android.app.usage.UsageStatsManager manager =
|
||||||
|
(android.app.usage.UsageStatsManager) getSystemService(Context.USAGE_STATS_SERVICE);
|
||||||
|
long endTime = System.currentTimeMillis();
|
||||||
|
long startTime = endTime - 24 * 3600 * 1000; // 近24小时
|
||||||
|
List<android.app.usage.UsageStats> statsList = manager.queryUsageStats(
|
||||||
|
android.app.usage.UsageStatsManager.INTERVAL_DAILY, startTime, endTime);
|
||||||
|
|
||||||
|
for (android.app.usage.UsageStats stats : statsList) {
|
||||||
|
long runTimeMs = stats.getTotalTimeInForeground();
|
||||||
|
String packageName = stats.getPackageName();
|
||||||
|
LogUtils.d(TAG, "包名" + packageName + "24小时运行时长:" + formatRunTime(runTimeMs));
|
||||||
|
runTimeMap.put(packageName, runTimeMs);
|
||||||
|
if (packageName.equals("aidepro.top")) {
|
||||||
|
LogUtils.d(TAG, String.format("runTimeMap.put(packageName, runTimeMs) 特殊查询 %s 查询有结果。", packageName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
LogUtils.e(TAG, "获取应用运行时长失败:" + e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LogUtils.d(TAG, String.format("应用运行时长列表数量%d。", runTimeMap.size()));
|
||||||
|
return runTimeMap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 格式化运行时长(逻辑不变)
|
||||||
|
*/
|
||||||
|
private String formatRunTime(long runTimeMs) {
|
||||||
|
if (runTimeMs <= 0) {
|
||||||
|
return "0秒";
|
||||||
|
}
|
||||||
|
long seconds = runTimeMs / 1000;
|
||||||
|
long hours = seconds / 3600;
|
||||||
|
long minutes = (seconds % 3600) / 60;
|
||||||
|
seconds = seconds % 60;
|
||||||
|
|
||||||
|
if (hours > 0) {
|
||||||
|
return String.format("%d时%d分%d秒", hours, minutes, seconds);
|
||||||
|
} else if (minutes > 0) {
|
||||||
|
return String.format("%d分%d秒", minutes, seconds);
|
||||||
|
} else {
|
||||||
|
return String.format("%d秒", seconds);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 权限检查(逻辑不变)
|
||||||
|
*/
|
||||||
|
private boolean hasUsageStatsPermission(Context context) {
|
||||||
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
android.app.usage.UsageStatsManager manager =
|
||||||
|
(android.app.usage.UsageStatsManager) context.getSystemService(Context.USAGE_STATS_SERVICE);
|
||||||
|
if (manager == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
long endTime = System.currentTimeMillis();
|
||||||
|
long startTime = endTime - 1000 * 60;
|
||||||
|
List<android.app.usage.UsageStats> statsList = manager.queryUsageStats(
|
||||||
|
android.app.usage.UsageStatsManager.INTERVAL_DAILY, startTime, endTime);
|
||||||
|
|
||||||
|
return statsList != null && !statsList.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 【核心修改】数据模型:明确字段含义
|
||||||
|
* - consumption:单次耗电(两次电池广播间的消耗,float类型便于计算)
|
||||||
|
* - totalConsumption:累计耗电(24小时初始化值+后续单次累加,显示用)
|
||||||
|
*/
|
||||||
|
public static class AppBatteryModel {
|
||||||
|
private String packageName; // 应用包名(核心标识)
|
||||||
|
private float consumption; // 单次耗电(mAh,float类型)
|
||||||
|
private float totalConsumption;// 累计耗电(mAh,显示+排序用)
|
||||||
|
private long runTime; // 运行时长(ms)
|
||||||
|
|
||||||
|
// Java7 显式构造:初始化单次耗电、累计耗电为0
|
||||||
|
public AppBatteryModel(String packageName, float consumption, float totalConsumption, long runTime) {
|
||||||
|
this.packageName = packageName;
|
||||||
|
this.consumption = consumption; // 单次耗电初始为0
|
||||||
|
this.totalConsumption = totalConsumption; // 累计耗电初始为0(后续初始化时赋值)
|
||||||
|
this.runTime = runTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Getter/Setter:覆盖所有字段,确保数据操作正常
|
||||||
|
public String getPackageName() {
|
||||||
|
return packageName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public float getConsumption() {
|
||||||
|
return consumption; // 获取单次耗电
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setConsumption(float consumption) {
|
||||||
|
this.consumption = consumption; // 设置单次耗电
|
||||||
|
}
|
||||||
|
|
||||||
|
public float getTotalConsumption() {
|
||||||
|
return totalConsumption; // 获取累计耗电(显示用)
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTotalConsumption(float totalConsumption) {
|
||||||
|
this.totalConsumption = totalConsumption; // 设置累计耗电(初始化/累加用)
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getRunTime() {
|
||||||
|
return runTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRunTime(long runTime) {
|
||||||
|
this.runTime = runTime;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* RecyclerView 适配器:仅显示累计耗电(totalConsumption),逻辑适配模型修改
|
||||||
|
*/
|
||||||
|
public static class BatteryReportAdapter extends RecyclerView.Adapter<BatteryReportAdapter.ViewHolder> {
|
||||||
|
private Context mContext;
|
||||||
|
private List<AppBatteryModel> mDataList;
|
||||||
|
private PackageManager mPm;
|
||||||
|
private Map<String, String> mPackageToNameCache;
|
||||||
|
|
||||||
|
// Java7 显式构造:接收名称缓存,确保显示时高效获取应用名
|
||||||
|
public BatteryReportAdapter(Context context, List<AppBatteryModel> dataList,
|
||||||
|
PackageManager pm, Map<String, String> packageToNameCache) {
|
||||||
|
this.mContext = context;
|
||||||
|
this.mDataList = dataList;
|
||||||
|
this.mPm = pm;
|
||||||
|
this.mPackageToNameCache = packageToNameCache;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
|
||||||
|
// 加载系统列表项布局(text1显示应用名,text2显示累计耗电+时长)
|
||||||
|
View itemView = LayoutInflater.from(mContext)
|
||||||
|
.inflate(android.R.layout.simple_list_item_2, parent, false);
|
||||||
|
return new ViewHolder(itemView);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onBindViewHolder(ViewHolder holder, int position) {
|
||||||
|
// Java7 显式非空判断:避免空指针异常
|
||||||
|
if (mDataList == null || mDataList.isEmpty() || position >= mDataList.size()) {
|
||||||
|
holder.tvAppName.setText("未知应用");
|
||||||
|
holder.tvConsumption.setText("累计耗电:0.0 mAh | 运行时长:0秒");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
AppBatteryModel model = mDataList.get(position);
|
||||||
|
String packageName = model.getPackageName();
|
||||||
|
String appName = "";
|
||||||
|
|
||||||
|
// 优先从缓存获取应用名:减少PackageManager调用,提升性能
|
||||||
|
if (mPackageToNameCache != null && mPackageToNameCache.containsKey(packageName)) {
|
||||||
|
appName = mPackageToNameCache.get(packageName);
|
||||||
|
} else {
|
||||||
|
// 缓存无数据时兜底查询,并同步更新缓存
|
||||||
|
try {
|
||||||
|
ApplicationInfo appInfo = mPm.getApplicationInfo(packageName, 0);
|
||||||
|
appName = mPm.getApplicationLabel(appInfo).toString();
|
||||||
|
if (mPackageToNameCache != null) {
|
||||||
|
mPackageToNameCache.put(packageName, appName);
|
||||||
|
}
|
||||||
|
} catch (PackageManager.NameNotFoundException e) {
|
||||||
|
appName = packageName; // 包名不存在时用包名兜底
|
||||||
|
LogUtils.e("Adapter", "包名" + packageName + "对应的应用未找到:" + e.getMessage());
|
||||||
|
} catch (Exception e) {
|
||||||
|
appName = packageName; // 其他异常时用包名兜底
|
||||||
|
LogUtils.e("Adapter", "查询应用名称失败(包名:" + packageName + "):" + e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 显示逻辑:仅展示累计耗电(totalConsumption),隐藏单次耗电
|
||||||
|
holder.tvAppName.setText(appName);
|
||||||
|
// 格式化运行时长 + 累计耗电(保留1位小数,提升可读性)
|
||||||
|
String runTimeStr = ((BatteryReportActivity) mContext).formatRunTime(model.getRunTime());
|
||||||
|
String totalConsumptionText = String.format("累计耗电:%.1f mAh | 运行时长:%s",
|
||||||
|
model.getTotalConsumption(), runTimeStr);
|
||||||
|
holder.tvConsumption.setText(totalConsumptionText);
|
||||||
|
|
||||||
|
// 显示优化:文字颜色区分(避免所有应用均标蓝,仅示例可按需修改)
|
||||||
|
holder.tvAppName.setTextColor(mContext.getResources().getColor(android.R.color.black));
|
||||||
|
holder.tvConsumption.setTextColor(mContext.getResources().getColor(android.R.color.darker_gray));
|
||||||
|
|
||||||
|
// 调整文字大小:适配手机屏幕,提升可读性
|
||||||
|
holder.tvAppName.setTextSize(16);
|
||||||
|
holder.tvConsumption.setTextSize(14);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取列表长度:Java7 三元运算符判断空值,避免空指针
|
||||||
|
@Override
|
||||||
|
public int getItemCount() {
|
||||||
|
return mDataList == null ? 0 : mDataList.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ViewHolder:绑定系统布局控件,与显示逻辑对应
|
||||||
|
*/
|
||||||
|
public static class ViewHolder extends RecyclerView.ViewHolder {
|
||||||
|
TextView tvAppName; // 显示应用名称
|
||||||
|
TextView tvConsumption; // 显示累计耗电 + 运行时长
|
||||||
|
|
||||||
|
// Java7 显式构造:绑定控件ID(系统布局固定ID:text1、text2)
|
||||||
|
public ViewHolder(View itemView) {
|
||||||
|
super(itemView);
|
||||||
|
tvAppName = (TextView) itemView.findViewById(android.R.id.text1);
|
||||||
|
tvConsumption = (TextView) itemView.findViewById(android.R.id.text2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
package cc.winboll.studio.powerbell.activities;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @Author ZhanGSKen<zhangsken@qq.com>
|
|
||||||
* @Date 2025/03/22 14:20:15
|
|
||||||
*/
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.os.Bundle;
|
|
||||||
import androidx.recyclerview.widget.DividerItemDecoration;
|
|
||||||
import androidx.recyclerview.widget.LinearLayoutManager;
|
|
||||||
import androidx.recyclerview.widget.RecyclerView;
|
|
||||||
import cc.winboll.studio.powerbell.R;
|
|
||||||
import cc.winboll.studio.powerbell.adapters.BatteryAdapter;
|
|
||||||
import cc.winboll.studio.powerbell.beans.BatteryData;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class BatteryReporterActivity extends Activity {
|
|
||||||
public static final String TAG = "BatteryReporterActivity";
|
|
||||||
|
|
||||||
private RecyclerView rvBatteryReport;
|
|
||||||
private BatteryAdapter adapter;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
|
||||||
super.onCreate(savedInstanceState);
|
|
||||||
setContentView(R.layout.activity_battery_reporter);
|
|
||||||
|
|
||||||
rvBatteryReport = findViewById(R.id.rvBatteryReport);
|
|
||||||
setupRecyclerView();
|
|
||||||
loadSampleData();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setupRecyclerView() {
|
|
||||||
adapter = new BatteryAdapter();
|
|
||||||
rvBatteryReport.setLayoutManager(new LinearLayoutManager(this));
|
|
||||||
rvBatteryReport.setAdapter(adapter);
|
|
||||||
rvBatteryReport.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL));
|
|
||||||
}
|
|
||||||
|
|
||||||
private void loadSampleData() {
|
|
||||||
List<BatteryData> dataList = Arrays.asList(
|
|
||||||
new BatteryData(95, "01:23:45", "00:05:12"),
|
|
||||||
new BatteryData(80, "02:15:30", "00:10:00"),
|
|
||||||
new BatteryData(65, "03:45:15", "00:15:30"),
|
|
||||||
new BatteryData(50, "05:00:00", "00:20:45")
|
|
||||||
);
|
|
||||||
adapter.updateData(dataList);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -4,11 +4,12 @@ import android.app.Activity;
|
|||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
|
import android.widget.Switch;
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
import cc.winboll.studio.libaes.views.AOHPCTCSeekBar;
|
import cc.winboll.studio.libaes.views.AOHPCTCSeekBar;
|
||||||
import cc.winboll.studio.libaes.views.AToolbar;
|
import cc.winboll.studio.libaes.views.AToolbar;
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
import cc.winboll.studio.libappbase.utils.ToastUtils;
|
import cc.winboll.studio.libappbase.ToastUtils;
|
||||||
import cc.winboll.studio.powerbell.App;
|
import cc.winboll.studio.powerbell.App;
|
||||||
import cc.winboll.studio.powerbell.R;
|
import cc.winboll.studio.powerbell.R;
|
||||||
import cc.winboll.studio.powerbell.beans.BatteryInfoBean;
|
import cc.winboll.studio.powerbell.beans.BatteryInfoBean;
|
||||||
@@ -16,7 +17,6 @@ import cc.winboll.studio.powerbell.receivers.ControlCenterServiceReceiver;
|
|||||||
import cc.winboll.studio.powerbell.utils.AppCacheUtils;
|
import cc.winboll.studio.powerbell.utils.AppCacheUtils;
|
||||||
import cc.winboll.studio.powerbell.utils.StringUtils;
|
import cc.winboll.studio.powerbell.utils.StringUtils;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import android.widget.Switch;
|
|
||||||
|
|
||||||
public class ClearRecordActivity extends Activity {
|
public class ClearRecordActivity extends Activity {
|
||||||
|
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ package cc.winboll.studio.powerbell.activities;
|
|||||||
*/
|
*/
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.app.Dialog;
|
import android.app.Dialog;
|
||||||
|
import android.content.Intent;
|
||||||
import android.graphics.Bitmap;
|
import android.graphics.Bitmap;
|
||||||
import android.graphics.BitmapFactory;
|
import android.graphics.BitmapFactory;
|
||||||
import android.graphics.Color;
|
import android.graphics.Color;
|
||||||
@@ -19,11 +20,11 @@ import android.widget.ImageView;
|
|||||||
import android.widget.RelativeLayout;
|
import android.widget.RelativeLayout;
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
import android.widget.Toast;
|
import android.widget.Toast;
|
||||||
|
import cc.winboll.studio.libaes.interfaces.IWinBoLLActivity;
|
||||||
import cc.winboll.studio.libaes.views.AToolbar;
|
import cc.winboll.studio.libaes.views.AToolbar;
|
||||||
import cc.winboll.studio.libappbase.GlobalApplication;
|
import cc.winboll.studio.libappbase.GlobalApplication;
|
||||||
import cc.winboll.studio.libappbase.winboll.IWinBoLLActivity;
|
|
||||||
import cc.winboll.studio.powerbell.MainActivity;
|
|
||||||
import cc.winboll.studio.powerbell.R;
|
import cc.winboll.studio.powerbell.R;
|
||||||
|
import cc.winboll.studio.powerbell.activities.BackgroundPictureActivity;
|
||||||
import cc.winboll.studio.powerbell.activities.PixelPickerActivity;
|
import cc.winboll.studio.powerbell.activities.PixelPickerActivity;
|
||||||
import cc.winboll.studio.powerbell.beans.BackgroundPictureBean;
|
import cc.winboll.studio.powerbell.beans.BackgroundPictureBean;
|
||||||
import cc.winboll.studio.powerbell.utils.BackgroundPictureUtils;
|
import cc.winboll.studio.powerbell.utils.BackgroundPictureUtils;
|
||||||
@@ -233,7 +234,10 @@ public class PixelPickerActivity extends WinBoLLActivity implements IWinBoLLActi
|
|||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
if (item.getItemId() == android.R.id.home) {
|
if (item.getItemId() == android.R.id.home) {
|
||||||
GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), BackgroundPictureActivity.class);
|
Intent intent = new Intent();
|
||||||
|
intent.setClass(this, BackgroundPictureActivity.class);
|
||||||
|
startActivity(intent);
|
||||||
|
//GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), );
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
// 在switch语句中处理每个ID,并在处理完后返回true,未处理的情况返回false。
|
// 在switch语句中处理每个ID,并在处理完后返回true,未处理的情况返回false。
|
||||||
@@ -243,7 +247,10 @@ public class PixelPickerActivity extends WinBoLLActivity implements IWinBoLLActi
|
|||||||
@Override
|
@Override
|
||||||
public void onBackPressed() {
|
public void onBackPressed() {
|
||||||
super.onBackPressed();
|
super.onBackPressed();
|
||||||
GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), BackgroundPictureActivity.class);
|
Intent intent = new Intent();
|
||||||
|
intent.setClass(this, BackgroundPictureActivity.class);
|
||||||
|
startActivity(intent);
|
||||||
|
//GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), BackgroundPictureActivity.class);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,40 +5,99 @@ package cc.winboll.studio.powerbell.activities;
|
|||||||
* @Date 2025/06/19 20:35
|
* @Date 2025/06/19 20:35
|
||||||
* @Describe 应用窗口基类
|
* @Describe 应用窗口基类
|
||||||
*/
|
*/
|
||||||
import android.app.Activity;
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.view.MenuItem;
|
import android.view.MenuItem;
|
||||||
import cc.winboll.studio.libappbase.GlobalApplication;
|
import androidx.appcompat.app.AppCompatActivity;
|
||||||
import cc.winboll.studio.libappbase.winboll.IWinBoLLActivity;
|
import cc.winboll.studio.libaes.interfaces.IWinBoLLActivity;
|
||||||
import cc.winboll.studio.powerbell.App;
|
import android.annotation.SuppressLint;
|
||||||
import cc.winboll.studio.powerbell.MainActivity;
|
import android.app.Activity;
|
||||||
|
import android.graphics.Color;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.util.TypedValue;
|
||||||
|
import android.view.Gravity;
|
||||||
|
import android.view.MenuItem;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.view.Window;
|
||||||
|
import android.view.WindowManager;
|
||||||
|
import android.widget.FrameLayout;
|
||||||
|
import android.widget.TextView;
|
||||||
|
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import androidx.appcompat.app.AppCompatActivity;
|
||||||
|
import androidx.appcompat.widget.Toolbar;
|
||||||
|
|
||||||
|
import com.miui.zeus.mimo.sdk.BuildConfig;
|
||||||
import cc.winboll.studio.powerbell.R;
|
import cc.winboll.studio.powerbell.R;
|
||||||
|
|
||||||
public abstract class WinBoLLActivity extends Activity implements IWinBoLLActivity {
|
@SuppressLint("SetTextI18n")
|
||||||
|
public abstract class WinBoLLActivity extends AppCompatActivity implements IWinBoLLActivity {
|
||||||
|
|
||||||
public static final String TAG = "WinBoLLActivity";
|
public static final String TAG = "WinBoLLActivity";
|
||||||
|
|
||||||
|
protected TextView mTagView;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
|
changeFullScreen(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStart() {
|
||||||
|
super.onStart();
|
||||||
|
addVersionNameToContentView();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void addVersionNameToContentView() {
|
||||||
|
if (!isTagViewVisible()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (mTagView == null) {
|
||||||
|
mTagView = new TextView(this);
|
||||||
|
mTagView.setTextColor(Color.GRAY);
|
||||||
|
mTagView.setTextSize(TypedValue.COMPLEX_UNIT_SP, 10);
|
||||||
|
mTagView.setText("MIMO SDK V" + BuildConfig.VERSION_NAME);
|
||||||
|
FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(
|
||||||
|
ViewGroup.LayoutParams.WRAP_CONTENT,
|
||||||
|
ViewGroup.LayoutParams.WRAP_CONTENT);
|
||||||
|
params.gravity = Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL;
|
||||||
|
FrameLayout frameLayout = findViewById(android.R.id.content);
|
||||||
|
frameLayout.addView(mTagView, params);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected boolean isTagViewVisible() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setupToolbar() {
|
||||||
|
Toolbar mToolbar = findViewById(R.id.toolbar);
|
||||||
|
if (mToolbar != null) {
|
||||||
|
setSupportActionBar(mToolbar);
|
||||||
|
if (getSupportActionBar() != null) {
|
||||||
|
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onPostCreate(Bundle savedInstanceState) {
|
protected void onPostCreate(Bundle savedInstanceState) {
|
||||||
super.onPostCreate(savedInstanceState);
|
super.onPostCreate(savedInstanceState);
|
||||||
GlobalApplication.getWinBoLLActivityManager().add(this);
|
//GlobalApplication.getWinBoLLActivityManager().add(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onDestroy() {
|
protected void onDestroy() {
|
||||||
super.onDestroy();
|
super.onDestroy();
|
||||||
GlobalApplication.getWinBoLLActivityManager().registeRemove(this);
|
//GlobalApplication.getWinBoLLActivityManager().registeRemove(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
if (item.getItemId() == android.R.id.home) {
|
if (item.getItemId() == android.R.id.home) {
|
||||||
GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), MainActivity.class);
|
//GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), MainActivity.class);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
// 在switch语句中处理每个ID,并在处理完后返回true,未处理的情况返回false。
|
// 在switch语句中处理每个ID,并在处理完后返回true,未处理的情况返回false。
|
||||||
@@ -48,6 +107,25 @@ public abstract class WinBoLLActivity extends Activity implements IWinBoLLActivi
|
|||||||
@Override
|
@Override
|
||||||
public void onBackPressed() {
|
public void onBackPressed() {
|
||||||
super.onBackPressed();
|
super.onBackPressed();
|
||||||
GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), MainActivity.class);
|
//GlobalApplication.getWinBoLLActivityManager().startWinBoLLActivity(getApplicationContext(), MainActivity.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void changeFullScreen(Activity activity) {
|
||||||
|
Window window = activity.getWindow();
|
||||||
|
if (window == null){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
View decorView = window.getDecorView();
|
||||||
|
if (decorView == null){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
int flag = decorView.getSystemUiVisibility();
|
||||||
|
flag |= View.SYSTEM_UI_FLAG_FULLSCREEN;
|
||||||
|
flag |= View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN;
|
||||||
|
flag |= View.SYSTEM_UI_FLAG_HIDE_NAVIGATION;
|
||||||
|
flag |= View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION;
|
||||||
|
flag |= View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY;
|
||||||
|
decorView.setSystemUiVisibility(flag);
|
||||||
|
window.setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,10 +14,12 @@ import android.content.ComponentName;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.ServiceConnection;
|
import android.content.ServiceConnection;
|
||||||
|
import android.os.Handler;
|
||||||
import android.os.IBinder;
|
import android.os.IBinder;
|
||||||
|
import android.os.Looper;
|
||||||
import android.widget.RemoteViews;
|
import android.widget.RemoteViews;
|
||||||
import cc.winboll.studio.libappbase.LogUtils;
|
import cc.winboll.studio.libappbase.LogUtils;
|
||||||
import cc.winboll.studio.libappbase.utils.ToastUtils;
|
import cc.winboll.studio.libappbase.ToastUtils;
|
||||||
import cc.winboll.studio.powerbell.App;
|
import cc.winboll.studio.powerbell.App;
|
||||||
import cc.winboll.studio.powerbell.MainActivity;
|
import cc.winboll.studio.powerbell.MainActivity;
|
||||||
import cc.winboll.studio.powerbell.R;
|
import cc.winboll.studio.powerbell.R;
|
||||||
@@ -32,8 +34,6 @@ import cc.winboll.studio.powerbell.utils.AppConfigUtils;
|
|||||||
import cc.winboll.studio.powerbell.utils.NotificationHelper;
|
import cc.winboll.studio.powerbell.utils.NotificationHelper;
|
||||||
import cc.winboll.studio.powerbell.utils.ServiceUtils;
|
import cc.winboll.studio.powerbell.utils.ServiceUtils;
|
||||||
import cc.winboll.studio.powerbell.utils.StringUtils;
|
import cc.winboll.studio.powerbell.utils.StringUtils;
|
||||||
import android.os.Handler;
|
|
||||||
import android.os.Looper;
|
|
||||||
|
|
||||||
public class ControlCenterService extends Service {
|
public class ControlCenterService extends Service {
|
||||||
|
|
||||||
@@ -73,8 +73,8 @@ public class ControlCenterService extends Service {
|
|||||||
mAppConfigUtils = App.getAppConfigUtils(this);
|
mAppConfigUtils = App.getAppConfigUtils(this);
|
||||||
mAppCacheUtils = App.getAppCacheUtils(this);
|
mAppCacheUtils = App.getAppCacheUtils(this);
|
||||||
mNotificationHelper = new NotificationHelper(ControlCenterService.this);
|
mNotificationHelper = new NotificationHelper(ControlCenterService.this);
|
||||||
|
|
||||||
|
|
||||||
if (mMyServiceConnection == null) {
|
if (mMyServiceConnection == null) {
|
||||||
mMyServiceConnection = new MyServiceConnection();
|
mMyServiceConnection = new MyServiceConnection();
|
||||||
}
|
}
|
||||||
@@ -105,7 +105,7 @@ public class ControlCenterService extends Service {
|
|||||||
Intent intent = new Intent(this, MainActivity.class);
|
Intent intent = new Intent(this, MainActivity.class);
|
||||||
notification = helper.showForegroundNotification(intent, getString(R.string.app_name), "Service Running, Click to open app");
|
notification = helper.showForegroundNotification(intent, getString(R.string.app_name), "Service Running, Click to open app");
|
||||||
startForeground(NotificationHelper.FOREGROUND_NOTIFICATION_ID, notification);
|
startForeground(NotificationHelper.FOREGROUND_NOTIFICATION_ID, notification);
|
||||||
|
|
||||||
// NotificationMessage notificationMessage=createNotificationMessage();
|
// NotificationMessage notificationMessage=createNotificationMessage();
|
||||||
// //Toast.makeText(getApplication(), "", Toast.LENGTH_SHORT).show();
|
// //Toast.makeText(getApplication(), "", Toast.LENGTH_SHORT).show();
|
||||||
// mNotificationUtils.createForegroundNotification(this, notificationMessage);
|
// mNotificationUtils.createForegroundNotification(this, notificationMessage);
|
||||||
@@ -116,7 +116,7 @@ public class ControlCenterService extends Service {
|
|||||||
mControlCenterServiceReceiver = new ControlCenterServiceReceiver(this);
|
mControlCenterServiceReceiver = new ControlCenterServiceReceiver(this);
|
||||||
mControlCenterServiceReceiver.registerAction(this);
|
mControlCenterServiceReceiver.registerAction(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
new Handler(Looper.getMainLooper()).postDelayed(new Runnable(){
|
new Handler(Looper.getMainLooper()).postDelayed(new Runnable(){
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -126,7 +126,7 @@ public class ControlCenterService extends Service {
|
|||||||
LogUtils.i(TAG, "Service Is Start.");
|
LogUtils.i(TAG, "Service Is Start.");
|
||||||
}
|
}
|
||||||
}, 2000);
|
}, 2000);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -263,15 +263,15 @@ public class ControlCenterService extends Service {
|
|||||||
NotificationHelper helper = new NotificationHelper(ControlCenterService.this);
|
NotificationHelper helper = new NotificationHelper(ControlCenterService.this);
|
||||||
Intent intent = new Intent(ControlCenterService.this, MainActivity.class);
|
Intent intent = new Intent(ControlCenterService.this, MainActivity.class);
|
||||||
helper.showTemporaryNotification(intent, getString(R.string.app_name), msg);
|
helper.showTemporaryNotification(intent, getString(R.string.app_name), msg);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// NotificationMessage notificationMessage = createNotificationMessage();
|
// NotificationMessage notificationMessage = createNotificationMessage();
|
||||||
// notificationMessage.setRemindMSG(szRemindMSG);
|
// notificationMessage.setRemindMSG(szRemindMSG);
|
||||||
// //LogUtils.d(TAG, "notificationMessage : " + notificationMessage.getRemindMSG());
|
// //LogUtils.d(TAG, "notificationMessage : " + notificationMessage.getRemindMSG());
|
||||||
// updateRemindNotification(notificationMessage);
|
// updateRemindNotification(notificationMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 设置颜色背景
|
// 设置颜色背景
|
||||||
public static RemoteViews setLinearLayoutColor(RemoteViews remoteViews, int viewId, int color) {
|
public static RemoteViews setLinearLayoutColor(RemoteViews remoteViews, int viewId, int color) {
|
||||||
remoteViews.setInt(viewId, "setBackgroundColor", color);
|
remoteViews.setInt(viewId, "setBackgroundColor", color);
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
package cc.winboll.studio.powerbell.utils;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.util.DisplayMetrics;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @Author ZhanGSKen&豆包大模型<zhangsken@qq.com>
|
||||||
|
* @Date 2025/11/14 11:14
|
||||||
|
* @Describe 米盟 MimoUtils
|
||||||
|
*/
|
||||||
|
public final class MimoUtils {
|
||||||
|
public static final String TAG = "Utils";
|
||||||
|
|
||||||
|
public static int dpToPx(Context context, float dp) {
|
||||||
|
DisplayMetrics displayMetrics = context.getResources().getDisplayMetrics();
|
||||||
|
return (int) (dp * displayMetrics.density + 0.5f);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int pxToDp(Context context, float px) {
|
||||||
|
DisplayMetrics displayMetrics = context.getResources().getDisplayMetrics();
|
||||||
|
return (int) (px / displayMetrics.density + 0.5f);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int pxToSp(Context context, float pxValue) {
|
||||||
|
DisplayMetrics displayMetrics = context.getResources().getDisplayMetrics();
|
||||||
|
return (int) (pxValue / displayMetrics.scaledDensity + 0.5f);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int spToPx(Context context, float spValue) {
|
||||||
|
DisplayMetrics displayMetrics = context.getResources().getDisplayMetrics();
|
||||||
|
return (int) (spValue * displayMetrics.scaledDensity + 0.5f);
|
||||||
|
}
|
||||||
|
}
|
||||||
30
powerbell/src/main/res/layout/activity_battery_report.xml
Normal file
30
powerbell/src/main/res/layout/activity_battery_report.xml
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<LinearLayout
|
||||||
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:background="@android:color/white">
|
||||||
|
|
||||||
|
<!-- 搜索框:提示文本改为“搜索应用名称或包名” -->
|
||||||
|
<EditText
|
||||||
|
android:id="@+id/et_search"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_margin="8dp"
|
||||||
|
android:padding="12dp"
|
||||||
|
android:hint="搜索应用名称或包名"
|
||||||
|
android:background="@android:drawable/btn_default_small"
|
||||||
|
android:inputType="text"
|
||||||
|
android:textSize="16sp"/>
|
||||||
|
|
||||||
|
<!-- 应用列表 -->
|
||||||
|
<androidx.recyclerview.widget.RecyclerView
|
||||||
|
android:id="@+id/rv_battery_report"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:layout_weight="1"
|
||||||
|
android:layout_marginLeft="8dp"
|
||||||
|
android:layout_marginRight="8dp"/>
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<LinearLayout
|
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:orientation="vertical"
|
|
||||||
android:padding="16dp">
|
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:text="电池使用报告"
|
|
||||||
android:textSize="24sp"
|
|
||||||
android:fontFamily="sans-serif-medium"
|
|
||||||
android:layout_marginBottom="16dp"/>
|
|
||||||
|
|
||||||
<androidx.recyclerview.widget.RecyclerView
|
|
||||||
android:id="@+id/rvBatteryReport"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:divider="@drawable/divider_line"
|
|
||||||
android:dividerHeight="1dp"/>
|
|
||||||
|
|
||||||
</LinearLayout>
|
|
||||||
@@ -6,7 +6,7 @@
|
|||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:orientation="vertical">
|
android:orientation="vertical">
|
||||||
|
|
||||||
<cc.winboll.studio.libaes.views.AToolbar
|
<cc.winboll.studio.libaes.views.ASupportToolbar
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="@dimen/toolbar_height"
|
android:layout_height="@dimen/toolbar_height"
|
||||||
android:id="@+id/toolbar"
|
android:id="@+id/toolbar"
|
||||||
@@ -22,14 +22,28 @@
|
|||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:id="@+id/activitymainRelativeLayout1"
|
android:id="@+id/activitymainRelativeLayout1"
|
||||||
android:background="#FFEE2121"/>
|
android:background="#FFB7B7B7"/>
|
||||||
|
|
||||||
<FrameLayout
|
<FrameLayout
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:id="@+id/activitymainFrameLayout1"/>
|
android:id="@+id/activitymainFrameLayout1"/>
|
||||||
|
|
||||||
|
|
||||||
</RelativeLayout>
|
</RelativeLayout>
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/activitymainLinearLayout1">
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:id="@+id/ads_container"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:orientation="vertical"/>
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<menu xmlns:android="http://schemas.android.com/apk/res/android">
|
<menu xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
<item
|
<item
|
||||||
android:id="@+id/action_battery_reporter"
|
android:id="@+id/action_battery_report"
|
||||||
android:title="@string/item_battery_reporter"/>
|
android:title="@string/item_battery_report"/>
|
||||||
<item
|
<item
|
||||||
android:id="@+id/action_clearrecord"
|
android:id="@+id/action_clearrecord"
|
||||||
android:title="@string/item_clearrecord"/>
|
android:title="@string/item_clearrecord"/>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<resources>
|
<resources>
|
||||||
<string name="app_name">能源钟</string>
|
<string name="app_name">PowerBell</string>
|
||||||
<string name="app_description">一个接收手机电量信息的应用,当电量值达到设定范围时会提醒用户。</string>
|
<string name="app_description">一个接收手机电量信息的应用,当电量值达到设定范围时会提醒用户。</string>
|
||||||
<string name="about_crashed">本应用崩溃了,作者水平有限,敬请谅解!</string>
|
<string name="about_crashed">本应用崩溃了,作者水平有限,敬请谅解!</string>
|
||||||
<string name="item_mainview">Main View</string>
|
<string name="item_mainview">Main View</string>
|
||||||
|
|||||||
@@ -25,16 +25,15 @@
|
|||||||
<color name="colorShuiDark">#FF0072A4</color>
|
<color name="colorShuiDark">#FF0072A4</color>
|
||||||
<color name="colorShuiAccent">#FF33C1FF</color>
|
<color name="colorShuiAccent">#FF33C1FF</color>
|
||||||
<color name="colorXinling">#FFFCC500</color>
|
<color name="colorXinling">#FFFCC500</color>
|
||||||
<color name="colorPrimary">@color/colorShuiDark</color>
|
<!--<color name="colorPrimary">@color/colorShuiDark</color>
|
||||||
<color name="colorPrimaryDark">@color/colorFeng</color>
|
<color name="colorPrimaryDark">@color/colorFeng</color>
|
||||||
<color name="colorAccent">@color/colorShui</color>
|
<color name="colorAccent">@color/colorShui</color>
|
||||||
<color name="colorYellow">#FFFFFF00</color>
|
-->
|
||||||
<color name="colorRed">#FFFF0000</color>
|
<color name="colorYellow">#FFBEBE48</color>
|
||||||
<color name="colorBlue">#FF000FFF</color>
|
<color name="colorRed">#FFC85C5C</color>
|
||||||
|
<color name="colorBlue">#FF2677C7</color>
|
||||||
<color name="colorBlack">#FF000000</color>
|
<color name="colorBlack">#FF000000</color>
|
||||||
<color name="colorUsege">@color/colorHuo</color>
|
<color name="colorText">@color/colorBlack</color>
|
||||||
<color name="colorCurrent">@color/colorShui</color>
|
|
||||||
<color name="colorCharge">@color/colorXinling</color>
|
|
||||||
<!-- 调试配置
|
<!-- 调试配置
|
||||||
<color name="colorYellow">#FF630066</color>
|
<color name="colorYellow">#FF630066</color>
|
||||||
<color name="colorRed">#FF23244D</color>
|
<color name="colorRed">#FF23244D</color>
|
||||||
@@ -47,15 +46,19 @@
|
|||||||
<color name="colorBlue">#FF000DE5</color>
|
<color name="colorBlue">#FF000DE5</color>
|
||||||
<color name="colorBlack">#FF000000</color>
|
<color name="colorBlack">#FF000000</color>
|
||||||
-->
|
-->
|
||||||
<!-- 原色配置
|
<!-- 原色配置 -->
|
||||||
<color name="colorYellow">#FFFFFF00</color>
|
<!--<color name="colorYellow">#FFFFFF00</color>
|
||||||
<color name="colorRed">#FFFF0000</color>
|
<color name="colorRed">#FFFF0000</color>
|
||||||
<color name="colorBlue">#FF000FFF</color>
|
<color name="colorBlue">#FF000FFF</color>
|
||||||
<color name="colorBlack">#FF000000</color>
|
<color name="colorBlack">#FF000000</color>
|
||||||
|
-->
|
||||||
|
<color name="colorPrimary">@color/colorShui</color>
|
||||||
|
<color name="colorPrimaryDark">@color/colorShuiDark</color>
|
||||||
|
<color name="colorAccent">@color/colorShuiAccent</color>
|
||||||
<color name="colorUsege">@color/colorRed</color>
|
<color name="colorUsege">@color/colorRed</color>
|
||||||
<color name="colorCurrent">@color/colorBlue</color>
|
<color name="colorCurrent">@color/colorBlue</color>
|
||||||
<color name="colorCharge">@color/colorYellow</color>
|
<color name="colorCharge">@color/colorYellow</color>
|
||||||
-->
|
|
||||||
<!--CustomSlideToUnlockView控件配置-->
|
<!--CustomSlideToUnlockView控件配置-->
|
||||||
<color name="colorCustomSlideToUnlockViewWhite">#FFFFFFFF</color>
|
<color name="colorCustomSlideToUnlockViewWhite">#FFFFFFFF</color>
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
<string name="about_crashed">This application has crashed, the author level is limited, please understand!</string>
|
<string name="about_crashed">This application has crashed, the author level is limited, please understand!</string>
|
||||||
<string name="item_mainview">Main View</string>
|
<string name="item_mainview">Main View</string>
|
||||||
<string name="item_aboutview">About</string>
|
<string name="item_aboutview">About</string>
|
||||||
<string name="item_battery_reporter">Battery Reporter</string>
|
<string name="item_battery_report">Battery Report</string>
|
||||||
<string name="item_clearrecord">Clear Record</string>
|
<string name="item_clearrecord">Clear Record</string>
|
||||||
<string name="item_changepicture">Change Picture</string>
|
<string name="item_changepicture">Change Picture</string>
|
||||||
<string name="item_devoloperoptionsview">Developer View</string>
|
<string name="item_devoloperoptionsview">Developer View</string>
|
||||||
|
|||||||
@@ -13,6 +13,9 @@
|
|||||||
<external-files-path
|
<external-files-path
|
||||||
name="external_file_path"
|
name="external_file_path"
|
||||||
path="." />
|
path="." />
|
||||||
|
<external-files-path
|
||||||
|
name="files_root"
|
||||||
|
path="mimoDownload" />
|
||||||
<!--代表app 外部存储区域根目录下的文件 Context.getExternalCacheDir目录下的目录-->
|
<!--代表app 外部存储区域根目录下的文件 Context.getExternalCacheDir目录下的目录-->
|
||||||
<external-cache-path
|
<external-cache-path
|
||||||
name="external_cache_path"
|
name="external_cache_path"
|
||||||
|
|||||||
11
powerbell/src/main/res/xml/network_security_config.xml
Normal file
11
powerbell/src/main/res/xml/network_security_config.xml
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<network-security-config>
|
||||||
|
<base-config cleartextTrafficPermitted="true" />
|
||||||
|
<debug-overrides>
|
||||||
|
<trust-anchors>
|
||||||
|
<!-- Trust user added CAs while debuggable only -->
|
||||||
|
<certificates src="system" />
|
||||||
|
<certificates src="user" />
|
||||||
|
</trust-anchors>
|
||||||
|
</debug-overrides>
|
||||||
|
</network-security-config>
|
||||||
Reference in New Issue
Block a user