Compare commits
3 Commits
positions-
...
regexputil
| Author | SHA1 | Date | |
|---|---|---|---|
| 452429ee9c | |||
|
|
a465c80ed7 | ||
|
|
fa462be666 |
@@ -1,8 +0,0 @@
|
||||
#Created by .winboll/winboll_app_build.gradle
|
||||
#Mon Sep 29 18:46:38 HKT 2025
|
||||
stageCount=3
|
||||
libraryProject=
|
||||
baseVersion=15.0
|
||||
publishVersion=15.0.2
|
||||
buildCount=0
|
||||
baseBetaVersion=15.0.3
|
||||
21
positions/proguard-rules.pro
vendored
@@ -1,21 +0,0 @@
|
||||
# Add project specific ProGuard rules here.
|
||||
# You can control the set of applied configuration files using the
|
||||
# proguardFiles setting in build.gradle.
|
||||
#
|
||||
# For more details, see
|
||||
# http://developer.android.com/guide/developing/tools/proguard.html
|
||||
|
||||
# If your project uses WebView with JS, uncomment the following
|
||||
# and specify the fully qualified class name to the JavaScript interface
|
||||
# class:
|
||||
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
|
||||
# public *;
|
||||
#}
|
||||
|
||||
# Uncomment this to preserve the line number information for
|
||||
# debugging stack traces.
|
||||
#-keepattributes SourceFile,LineNumberTable
|
||||
|
||||
# If you keep the line number information, uncomment this to
|
||||
# hide the original source file name.
|
||||
#-renamesourcefileattribute SourceFile
|
||||
@@ -1,55 +0,0 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<manifest
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
package="cc.winboll.studio.positions">
|
||||
|
||||
<!-- 1. 定位必需权限声明 -->
|
||||
<!-- 精确定位权限(GPS+网络定位,核心) -->
|
||||
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
|
||||
<!-- 粗略定位权限(仅安卓12+需要,兼容低版本可加) -->
|
||||
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
||||
<!-- 网络权限(可选,用于网络定位,提升室内定位精度) -->
|
||||
<uses-permission android:name="android.permission.INTERNET" />
|
||||
|
||||
<!-- 2. 声明定位硬件支持(可选,告诉系统应用需要定位功能) -->
|
||||
<uses-feature
|
||||
android:name="android.hardware.location.gps"
|
||||
android:required="false" /> <!-- false=无GPS也能使用(网络定位) -->
|
||||
|
||||
<application
|
||||
android:allowBackup="true"
|
||||
android:icon="@drawable/ic_launcher"
|
||||
android:label="@string/app_name"
|
||||
android:theme="@style/MyAppTheme"
|
||||
android:resizeableActivity="true"
|
||||
android:name=".App">
|
||||
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:label="@string/app_name">
|
||||
|
||||
<intent-filter>
|
||||
|
||||
<action android:name="android.intent.action.MAIN"/>
|
||||
|
||||
<category android:name="android.intent.category.LAUNCHER"/>
|
||||
|
||||
</intent-filter>
|
||||
|
||||
</activity>
|
||||
|
||||
<meta-data
|
||||
android:name="android.max_aspect"
|
||||
android:value="4.0"/>
|
||||
|
||||
<activity android:name=".GlobalApplication$CrashActivity"/>
|
||||
|
||||
<activity android:name="cc.winboll.studio.positions.activities.LocationActivity"/>
|
||||
|
||||
<!-- 4. 谷歌定位服务版本声明(避免版本兼容问题) -->
|
||||
<meta-data
|
||||
android:name="com.google.android.gms.version"
|
||||
android:value="@integer/google_play_services_version" />
|
||||
</application>
|
||||
|
||||
</manifest>
|
||||
@@ -1,345 +0,0 @@
|
||||
package cc.winboll.studio.positions;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.content.ClipData;
|
||||
import android.content.ClipboardManager;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.PackageInfo;
|
||||
import android.content.res.Resources;
|
||||
import android.graphics.Typeface;
|
||||
import android.os.Build;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.Looper;
|
||||
import android.text.TextUtils;
|
||||
import android.util.Log;
|
||||
import android.view.Gravity;
|
||||
import android.view.Menu;
|
||||
import android.view.MenuItem;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.HorizontalScrollView;
|
||||
import android.widget.ScrollView;
|
||||
import android.widget.TextView;
|
||||
import android.widget.Toast;
|
||||
import cc.winboll.studio.libappbase.GlobalApplication;
|
||||
import com.hjq.toast.ToastUtils;
|
||||
import com.hjq.toast.style.WhiteToastStyle;
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.Closeable;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.OutputStream;
|
||||
import java.lang.Thread.UncaughtExceptionHandler;
|
||||
import java.text.DateFormat;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Arrays;
|
||||
import java.util.Date;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
|
||||
public class App extends GlobalApplication {
|
||||
|
||||
private static Handler MAIN_HANDLER = new Handler(Looper.getMainLooper());
|
||||
|
||||
@Override
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
|
||||
// 初始化 Toast 框架
|
||||
ToastUtils.init(this);
|
||||
// 设置 Toast 布局样式
|
||||
//ToastUtils.setView(R.layout.view_toast);
|
||||
ToastUtils.setStyle(new WhiteToastStyle());
|
||||
ToastUtils.setGravity(Gravity.BOTTOM, 0, 200);
|
||||
|
||||
//CrashHandler.getInstance().registerGlobal(this);
|
||||
//CrashHandler.getInstance().registerPart(this);
|
||||
}
|
||||
|
||||
public static void write(InputStream input, OutputStream output) throws IOException {
|
||||
byte[] buf = new byte[1024 * 8];
|
||||
int len;
|
||||
while ((len = input.read(buf)) != -1) {
|
||||
output.write(buf, 0, len);
|
||||
}
|
||||
}
|
||||
|
||||
public static void write(File file, byte[] data) throws IOException {
|
||||
File parent = file.getParentFile();
|
||||
if (parent != null && !parent.exists()) parent.mkdirs();
|
||||
|
||||
ByteArrayInputStream input = new ByteArrayInputStream(data);
|
||||
FileOutputStream output = new FileOutputStream(file);
|
||||
try {
|
||||
write(input, output);
|
||||
} finally {
|
||||
closeIO(input, output);
|
||||
}
|
||||
}
|
||||
|
||||
public static String toString(InputStream input) throws IOException {
|
||||
ByteArrayOutputStream output = new ByteArrayOutputStream();
|
||||
write(input, output);
|
||||
try {
|
||||
return output.toString("UTF-8");
|
||||
} finally {
|
||||
closeIO(input, output);
|
||||
}
|
||||
}
|
||||
|
||||
public static void closeIO(Closeable... closeables) {
|
||||
for (Closeable closeable : closeables) {
|
||||
try {
|
||||
if (closeable != null) closeable.close();
|
||||
} catch (IOException ignored) {}
|
||||
}
|
||||
}
|
||||
|
||||
public static class CrashHandler {
|
||||
|
||||
public static final UncaughtExceptionHandler DEFAULT_UNCAUGHT_EXCEPTION_HANDLER = Thread.getDefaultUncaughtExceptionHandler();
|
||||
|
||||
private static CrashHandler sInstance;
|
||||
|
||||
private PartCrashHandler mPartCrashHandler;
|
||||
|
||||
public static CrashHandler getInstance() {
|
||||
if (sInstance == null) {
|
||||
sInstance = new CrashHandler();
|
||||
}
|
||||
return sInstance;
|
||||
}
|
||||
|
||||
public void registerGlobal(Context context) {
|
||||
registerGlobal(context, null);
|
||||
}
|
||||
|
||||
public void registerGlobal(Context context, String crashDir) {
|
||||
Thread.setDefaultUncaughtExceptionHandler(new UncaughtExceptionHandlerImpl(context.getApplicationContext(), crashDir));
|
||||
}
|
||||
|
||||
public void unregister() {
|
||||
Thread.setDefaultUncaughtExceptionHandler(DEFAULT_UNCAUGHT_EXCEPTION_HANDLER);
|
||||
}
|
||||
|
||||
public void registerPart(Context context) {
|
||||
unregisterPart(context);
|
||||
mPartCrashHandler = new PartCrashHandler(context.getApplicationContext());
|
||||
MAIN_HANDLER.postAtFrontOfQueue(mPartCrashHandler);
|
||||
}
|
||||
|
||||
public void unregisterPart(Context context) {
|
||||
if (mPartCrashHandler != null) {
|
||||
mPartCrashHandler.isRunning.set(false);
|
||||
mPartCrashHandler = null;
|
||||
}
|
||||
}
|
||||
|
||||
private static class PartCrashHandler implements Runnable {
|
||||
|
||||
private final Context mContext;
|
||||
|
||||
public AtomicBoolean isRunning = new AtomicBoolean(true);
|
||||
|
||||
public PartCrashHandler(Context context) {
|
||||
this.mContext = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
while (isRunning.get()) {
|
||||
try {
|
||||
Looper.loop();
|
||||
} catch (final Throwable e) {
|
||||
e.printStackTrace();
|
||||
if (isRunning.get()) {
|
||||
MAIN_HANDLER.post(new Runnable(){
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
Toast.makeText(mContext, e.toString(), Toast.LENGTH_LONG).show();
|
||||
}
|
||||
});
|
||||
} else {
|
||||
if (e instanceof RuntimeException) {
|
||||
throw (RuntimeException)e;
|
||||
} else {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static class UncaughtExceptionHandlerImpl implements UncaughtExceptionHandler {
|
||||
|
||||
private static DateFormat DATE_FORMAT = new SimpleDateFormat("yyyy_MM_dd-HH_mm_ss");
|
||||
|
||||
private final Context mContext;
|
||||
|
||||
private final File mCrashDir;
|
||||
|
||||
public UncaughtExceptionHandlerImpl(Context context, String crashDir) {
|
||||
this.mContext = context;
|
||||
this.mCrashDir = TextUtils.isEmpty(crashDir) ? new File(mContext.getExternalCacheDir(), "crash") : new File(crashDir);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void uncaughtException(Thread thread, Throwable throwable) {
|
||||
try {
|
||||
|
||||
String log = buildLog(throwable);
|
||||
writeLog(log);
|
||||
|
||||
try {
|
||||
Intent intent = new Intent(mContext, CrashActivity.class);
|
||||
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
||||
intent.putExtra(Intent.EXTRA_TEXT, log);
|
||||
mContext.startActivity(intent);
|
||||
} catch (Throwable e) {
|
||||
e.printStackTrace();
|
||||
writeLog(e.toString());
|
||||
}
|
||||
|
||||
throwable.printStackTrace();
|
||||
android.os.Process.killProcess(android.os.Process.myPid());
|
||||
System.exit(0);
|
||||
|
||||
} catch (Throwable e) {
|
||||
if (DEFAULT_UNCAUGHT_EXCEPTION_HANDLER != null) DEFAULT_UNCAUGHT_EXCEPTION_HANDLER.uncaughtException(thread, throwable);
|
||||
}
|
||||
}
|
||||
|
||||
private String buildLog(Throwable throwable) {
|
||||
String time = DATE_FORMAT.format(new Date());
|
||||
|
||||
String versionName = "unknown";
|
||||
long versionCode = 0;
|
||||
try {
|
||||
PackageInfo packageInfo = mContext.getPackageManager().getPackageInfo(mContext.getPackageName(), 0);
|
||||
versionName = packageInfo.versionName;
|
||||
versionCode = Build.VERSION.SDK_INT >= 28 ? packageInfo.getLongVersionCode() : packageInfo.versionCode;
|
||||
} catch (Throwable ignored) {}
|
||||
|
||||
LinkedHashMap<String, String> head = new LinkedHashMap<String, String>();
|
||||
head.put("Time Of Crash", time);
|
||||
head.put("Device", String.format("%s, %s", Build.MANUFACTURER, Build.MODEL));
|
||||
head.put("Android Version", String.format("%s (%d)", Build.VERSION.RELEASE, Build.VERSION.SDK_INT));
|
||||
head.put("App Version", String.format("%s (%d)", versionName, versionCode));
|
||||
head.put("Kernel", getKernel());
|
||||
head.put("Support Abis", Build.VERSION.SDK_INT >= 21 && Build.SUPPORTED_ABIS != null ? Arrays.toString(Build.SUPPORTED_ABIS): "unknown");
|
||||
head.put("Fingerprint", Build.FINGERPRINT);
|
||||
|
||||
StringBuilder builder = new StringBuilder();
|
||||
|
||||
for (String key : head.keySet()) {
|
||||
if (builder.length() != 0) builder.append("\n");
|
||||
builder.append(key);
|
||||
builder.append(" : ");
|
||||
builder.append(head.get(key));
|
||||
}
|
||||
|
||||
builder.append("\n\n");
|
||||
builder.append(Log.getStackTraceString(throwable));
|
||||
|
||||
return builder.toString();
|
||||
}
|
||||
|
||||
private void writeLog(String log) {
|
||||
String time = DATE_FORMAT.format(new Date());
|
||||
File file = new File(mCrashDir, "crash_" + time + ".txt");
|
||||
try {
|
||||
write(file, log.getBytes("UTF-8"));
|
||||
} catch (Throwable e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
private static String getKernel() {
|
||||
try {
|
||||
return App.toString(new FileInputStream("/proc/version")).trim();
|
||||
} catch (Throwable e) {
|
||||
return e.getMessage();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static final class CrashActivity extends Activity {
|
||||
|
||||
private String mLog;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
setTheme(android.R.style.Theme_DeviceDefault);
|
||||
setTitle("App Crash");
|
||||
|
||||
mLog = getIntent().getStringExtra(Intent.EXTRA_TEXT);
|
||||
|
||||
ScrollView contentView = new ScrollView(this);
|
||||
contentView.setFillViewport(true);
|
||||
|
||||
HorizontalScrollView horizontalScrollView = new HorizontalScrollView(this);
|
||||
|
||||
TextView textView = new TextView(this);
|
||||
int padding = dp2px(16);
|
||||
textView.setPadding(padding, padding, padding, padding);
|
||||
textView.setText(mLog);
|
||||
textView.setTextIsSelectable(true);
|
||||
textView.setTypeface(Typeface.DEFAULT);
|
||||
textView.setLinksClickable(true);
|
||||
|
||||
horizontalScrollView.addView(textView);
|
||||
contentView.addView(horizontalScrollView, ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
|
||||
|
||||
setContentView(contentView);
|
||||
}
|
||||
|
||||
private void restart() {
|
||||
Intent intent = getPackageManager().getLaunchIntentForPackage(getPackageName());
|
||||
if (intent != null) {
|
||||
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
||||
startActivity(intent);
|
||||
}
|
||||
finish();
|
||||
android.os.Process.killProcess(android.os.Process.myPid());
|
||||
System.exit(0);
|
||||
}
|
||||
|
||||
private static int dp2px(float dpValue) {
|
||||
final float scale = Resources.getSystem().getDisplayMetrics().density;
|
||||
return (int) (dpValue * scale + 0.5f);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu) {
|
||||
menu.add(0, android.R.id.copy, 0, android.R.string.copy)
|
||||
.setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM);
|
||||
return super.onCreateOptionsMenu(menu);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onOptionsItemSelected(MenuItem item) {
|
||||
switch (item.getItemId()) {
|
||||
case android.R.id.copy:
|
||||
ClipboardManager cm = (ClipboardManager) getSystemService(Context.CLIPBOARD_SERVICE);
|
||||
cm.setPrimaryClip(ClipData.newPlainText(getPackageName(), mLog));
|
||||
return true;
|
||||
}
|
||||
return super.onOptionsItemSelected(item);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
restart();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,38 +0,0 @@
|
||||
package cc.winboll.studio.positions;
|
||||
|
||||
import android.os.Bundle;
|
||||
import androidx.appcompat.app.AppCompatActivity;
|
||||
import androidx.appcompat.widget.Toolbar;
|
||||
import cc.winboll.studio.libappbase.LogView;
|
||||
import com.hjq.toast.ToastUtils;
|
||||
import android.view.View;
|
||||
import cc.winboll.studio.positions.activities.LocationActivity;
|
||||
import android.content.Intent;
|
||||
|
||||
public class MainActivity extends AppCompatActivity {
|
||||
|
||||
LogView mLogView;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_main);
|
||||
|
||||
Toolbar toolbar=(Toolbar)findViewById(R.id.toolbar);
|
||||
setSupportActionBar(toolbar);
|
||||
|
||||
mLogView = findViewById(R.id.logview);
|
||||
|
||||
ToastUtils.show("onCreate");
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onResume() {
|
||||
super.onResume();
|
||||
mLogView.start();
|
||||
}
|
||||
|
||||
public void onPositions(View view) {
|
||||
startActivity(new Intent(this, LocationActivity.class));
|
||||
}
|
||||
}
|
||||
@@ -1,219 +0,0 @@
|
||||
package cc.winboll.studio.positions.activities;
|
||||
|
||||
|
||||
/**
|
||||
* @Author ZhanGSKen&豆包大模型<zhangsken@qq.com>
|
||||
* @Date 2025/09/29 18:22
|
||||
* @Describe 当前位置实时显示
|
||||
*/
|
||||
|
||||
|
||||
import android.Manifest;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.location.Location;
|
||||
import android.os.Bundle;
|
||||
import android.widget.TextView;
|
||||
import android.widget.Toast;
|
||||
import androidx.annotation.NonNull;
|
||||
import androidx.appcompat.app.AppCompatActivity;
|
||||
import androidx.core.app.ActivityCompat;
|
||||
import com.google.android.gms.location.FusedLocationProviderClient;
|
||||
import com.google.android.gms.location.LocationCallback;
|
||||
import com.google.android.gms.location.LocationRequest;
|
||||
import com.google.android.gms.location.LocationResult;
|
||||
import com.google.android.gms.location.LocationServices;
|
||||
import com.google.android.gms.tasks.OnSuccessListener;
|
||||
import cc.winboll.studio.positions.R;
|
||||
|
||||
|
||||
/**
|
||||
* 实时定位活动窗口:
|
||||
* 1. 申请定位必需权限(精确定位+粗略定位)
|
||||
* 2. 初始化FusedLocationProviderClient(谷歌官方定位服务,兼容所有安卓版本)
|
||||
* 3. 实时监听位置变化,更新显示经度、纬度
|
||||
*/
|
||||
public class LocationActivity extends AppCompatActivity {
|
||||
|
||||
public static final String TAG = "LocationActivity";
|
||||
|
||||
// 1. 核心组件与常量定义
|
||||
private static final int REQUEST_LOCATION_PERMISSIONS = 1004; // 定位权限请求码
|
||||
private FusedLocationProviderClient fusedLocationClient; // 定位核心客户端
|
||||
private LocationCallback locationCallback; // 位置变化监听器
|
||||
private LocationRequest locationRequest; // 定位请求配置(频率、精度等)
|
||||
|
||||
// UI控件:用于显示经纬度(需在布局中定义对应ID)
|
||||
private TextView tvLongitude; // 经度显示
|
||||
private TextView tvLatitude; // 纬度显示
|
||||
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
// 2. 加载布局(需手动创建对应布局文件,见下方说明)
|
||||
setContentView(R.layout.activity_location);
|
||||
|
||||
// 3. 绑定UI控件(与布局文件中的TextView ID对应)
|
||||
tvLongitude = findViewById(R.id.tv_longitude);
|
||||
tvLatitude = findViewById(R.id.tv_latitude);
|
||||
|
||||
// 4. 初始化定位相关组件
|
||||
initLocationConfig();
|
||||
|
||||
// 5. 检查并申请定位权限(无权限则申请,有权限则直接启动定位)
|
||||
if (checkLocationPermissions()) {
|
||||
startRealTimeLocation(); // 权限已授予 → 启动实时定位
|
||||
} else {
|
||||
requestLocationPermissions(); // 权限未授予 → 申请权限
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 初始化定位配置:设置定位精度、更新频率等
|
||||
*/
|
||||
private void initLocationConfig() {
|
||||
// 初始化定位客户端(谷歌官方推荐,替代旧的LocationManager)
|
||||
fusedLocationClient = LocationServices.getFusedLocationProviderClient(this);
|
||||
|
||||
// 配置定位请求:实时更新(1秒请求一次,最小间隔500毫秒,最高精度)
|
||||
locationRequest = new LocationRequest.Builder(1000) // 定位更新间隔(毫秒)
|
||||
.setMinUpdateIntervalMillis(500) // 最小更新间隔(避免频繁更新耗电)
|
||||
.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY) // 高精度定位(优先GPS)
|
||||
.build();
|
||||
|
||||
// 初始化位置变化监听器:位置更新时触发(实时更新UI)
|
||||
locationCallback = new LocationCallback() {
|
||||
@Override
|
||||
public void onLocationResult(@NonNull LocationResult locationResult) {
|
||||
super.onLocationResult(locationResult);
|
||||
// 获取最新位置信息(locationResult包含最近一次或多次位置)
|
||||
Location latestLocation = locationResult.getLastLocation();
|
||||
if (latestLocation != null) {
|
||||
// 6. 提取经纬度并更新UI(实时显示)
|
||||
double longitude = latestLocation.getLongitude(); // 经度(东经为正,西经为负)
|
||||
double latitude = latestLocation.getLatitude(); // 纬度(北纬为正,南纬为负)
|
||||
|
||||
// 更新TextView显示(保留6位小数,精度足够日常使用)
|
||||
tvLongitude.setText(String.format("当前经度:%.6f", longitude));
|
||||
tvLatitude.setText(String.format("当前纬度:%.6f", latitude));
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 检查定位必需权限是否已授予(精确定位+粗略定位,覆盖所有安卓版本)
|
||||
*/
|
||||
private boolean checkLocationPermissions() {
|
||||
// 安卓12+(API31+)新增精确定位权限,需同时检查2个权限;低版本只需检查1个
|
||||
/*if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.S) {
|
||||
return ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED
|
||||
&& ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) == PackageManager.PERMISSION_GRANTED;
|
||||
} else {*/
|
||||
return ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED;
|
||||
//}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 申请定位必需权限(弹系统授权弹窗)
|
||||
*/
|
||||
private void requestLocationPermissions() {
|
||||
/*if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.S) {
|
||||
// 安卓12+:同时申请精确定位+粗略定位
|
||||
ActivityCompat.requestPermissions(
|
||||
this,
|
||||
new String[]{Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.ACCESS_COARSE_LOCATION},
|
||||
REQUEST_LOCATION_PERMISSIONS
|
||||
);
|
||||
} else {*/
|
||||
// 安卓12以下:仅申请精确定位(包含粗略定位能力)
|
||||
ActivityCompat.requestPermissions(
|
||||
this,
|
||||
new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
|
||||
REQUEST_LOCATION_PERMISSIONS
|
||||
);
|
||||
//}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 启动实时定位:注册位置监听器,开始接收位置更新
|
||||
*/
|
||||
private void startRealTimeLocation() {
|
||||
// 权限兜底检查(避免异常)
|
||||
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
|
||||
Toast.makeText(this, "定位权限未授予,无法启动定位", Toast.LENGTH_SHORT).show();
|
||||
return;
|
||||
}
|
||||
|
||||
// 1. 先获取一次当前位置(快速显示初始经纬度)
|
||||
fusedLocationClient.getLastLocation()
|
||||
.addOnSuccessListener(this, new OnSuccessListener<Location>() {
|
||||
@Override
|
||||
public void onSuccess(Location location) {
|
||||
if (location != null) {
|
||||
// 显示初始经纬度
|
||||
tvLongitude.setText(String.format("当前经度:%.6f", location.getLongitude()));
|
||||
tvLatitude.setText(String.format("当前纬度:%.6f", location.getLatitude()));
|
||||
} else {
|
||||
// 无历史位置(如首次启动),提示“等待定位更新”
|
||||
tvLongitude.setText("当前经度:等待更新...");
|
||||
tvLatitude.setText("当前纬度:等待更新...");
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// 2. 注册监听器,接收实时位置更新(持续监听)
|
||||
fusedLocationClient.requestLocationUpdates(
|
||||
locationRequest,
|
||||
locationCallback,
|
||||
getMainLooper() // 在主线程更新UI(避免线程异常)
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 处理权限申请结果(用户同意/拒绝后触发)
|
||||
*/
|
||||
@Override
|
||||
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
|
||||
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
|
||||
if (requestCode == REQUEST_LOCATION_PERMISSIONS) {
|
||||
// 检查是否所有必需权限都已授予
|
||||
boolean allGranted = true;
|
||||
for (int result : grantResults) {
|
||||
if (result != PackageManager.PERMISSION_GRANTED) {
|
||||
allGranted = false;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (allGranted) {
|
||||
// 权限同意 → 启动实时定位
|
||||
startRealTimeLocation();
|
||||
} else {
|
||||
// 权限拒绝 → 提示用户(无法定位)
|
||||
Toast.makeText(this, "定位权限被拒绝,无法显示位置信息", Toast.LENGTH_SHORT).show();
|
||||
tvLongitude.setText("当前经度:无权限");
|
||||
tvLatitude.setText("当前纬度:无权限");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 活动销毁时:停止定位监听(避免内存泄漏、减少耗电)
|
||||
*/
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// 移除定位监听器(核心:防止Activity销毁后仍在监听,导致内存泄漏)
|
||||
if (fusedLocationClient != null && locationCallback != null) {
|
||||
fusedLocationClient.removeLocationUpdates(locationCallback);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,34 +0,0 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:aapt="http://schemas.android.com/aapt"
|
||||
android:width="108dp"
|
||||
android:height="108dp"
|
||||
android:viewportHeight="108"
|
||||
android:viewportWidth="108">
|
||||
<path
|
||||
android:fillType="evenOdd"
|
||||
android:pathData="M32,64C32,64 38.39,52.99 44.13,50.95C51.37,48.37 70.14,49.57 70.14,49.57L108.26,87.69L108,109.01L75.97,107.97L32,64Z"
|
||||
android:strokeColor="#00000000"
|
||||
android:strokeWidth="1">
|
||||
<aapt:attr name="android:fillColor">
|
||||
<gradient
|
||||
android:endX="78.5885"
|
||||
android:endY="90.9159"
|
||||
android:startX="48.7653"
|
||||
android:startY="61.0927"
|
||||
android:type="linear">
|
||||
<item
|
||||
android:color="#44000000"
|
||||
android:offset="0.0" />
|
||||
<item
|
||||
android:color="#00000000"
|
||||
android:offset="1.0" />
|
||||
</gradient>
|
||||
</aapt:attr>
|
||||
</path>
|
||||
<path
|
||||
android:fillColor="#FFFFFF"
|
||||
android:fillType="nonZero"
|
||||
android:pathData="M66.94,46.02L66.94,46.02C72.44,50.07 76,56.61 76,64L32,64C32,56.61 35.56,50.11 40.98,46.06L36.18,41.19C35.45,40.45 35.45,39.3 36.18,38.56C36.91,37.81 38.05,37.81 38.78,38.56L44.25,44.05C47.18,42.57 50.48,41.71 54,41.71C57.48,41.71 60.78,42.57 63.68,44.05L69.11,38.56C69.84,37.81 70.98,37.81 71.71,38.56C72.44,39.3 72.44,40.45 71.71,41.19L66.94,46.02ZM62.94,56.92C64.08,56.92 65,56.01 65,54.88C65,53.76 64.08,52.85 62.94,52.85C61.8,52.85 60.88,53.76 60.88,54.88C60.88,56.01 61.8,56.92 62.94,56.92ZM45.06,56.92C46.2,56.92 47.13,56.01 47.13,54.88C47.13,53.76 46.2,52.85 45.06,52.85C43.92,52.85 43,53.76 43,54.88C43,56.01 43.92,56.92 45.06,56.92Z"
|
||||
android:strokeColor="#00000000"
|
||||
android:strokeWidth="1" />
|
||||
</vector>
|
||||
|
Before Width: | Height: | Size: 2.2 MiB |
@@ -1,34 +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:gravity="center_horizontal|center_vertical"
|
||||
android:orientation="vertical"
|
||||
android:padding="20dp">
|
||||
|
||||
<!-- 标题 -->
|
||||
<TextView
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:text="实时位置信息"
|
||||
android:textSize="22sp"
|
||||
android:textStyle="bold"/>
|
||||
|
||||
<!-- 经度显示(大字体,清晰可见) -->
|
||||
<TextView
|
||||
android:id="@+id/tv_longitude"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:text="当前经度:等待更新..."
|
||||
android:textSize="18sp"/>
|
||||
|
||||
<!-- 纬度显示(大字体,清晰可见) -->
|
||||
<TextView
|
||||
android:id="@+id/tv_latitude"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:text="当前纬度:等待更新..."
|
||||
android:textSize="18sp"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
@@ -1,51 +0,0 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<LinearLayout
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:orientation="vertical">
|
||||
|
||||
<com.google.android.material.appbar.AppBarLayout
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
|
||||
|
||||
<androidx.appcompat.widget.Toolbar
|
||||
android:id="@+id/toolbar"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="?attr/actionBarSize"
|
||||
app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>
|
||||
|
||||
</com.google.android.material.appbar.AppBarLayout>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="vertical"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="0dp"
|
||||
android:layout_weight="1.0"
|
||||
android:gravity="center_vertical|center_horizontal">
|
||||
|
||||
<Button
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:text="Positions"
|
||||
android:onClick="onPositions"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="vertical"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="0dp"
|
||||
android:layout_weight="1.0">
|
||||
|
||||
<cc.winboll.studio.libappbase.LogView
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:id="@+id/logview"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<background android:drawable="@drawable/ic_launcher_background" />
|
||||
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||
</adaptive-icon>
|
||||
@@ -1,5 +0,0 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<background android:drawable="@drawable/ic_launcher_background" />
|
||||
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||
</adaptive-icon>
|
||||
|
Before Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 4.9 KiB |
|
Before Width: | Height: | Size: 2.0 KiB |
|
Before Width: | Height: | Size: 2.8 KiB |
|
Before Width: | Height: | Size: 4.5 KiB |
|
Before Width: | Height: | Size: 6.9 KiB |
|
Before Width: | Height: | Size: 6.3 KiB |
|
Before Width: | Height: | Size: 10 KiB |
|
Before Width: | Height: | Size: 9.0 KiB |
|
Before Width: | Height: | Size: 15 KiB |
@@ -1,6 +0,0 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<color name="colorPrimary">#009688</color>
|
||||
<color name="colorPrimaryDark">#00796B</color>
|
||||
<color name="colorAccent">#FF9800</color>
|
||||
</resources>
|
||||
@@ -1,4 +0,0 @@
|
||||
<resources>
|
||||
<string name="app_name">Positions</string>
|
||||
|
||||
</resources>
|
||||
@@ -1,11 +0,0 @@
|
||||
<resources>
|
||||
|
||||
<!-- Base application theme. -->
|
||||
<style name="MyAppTheme" parent="Theme.AppCompat.Light.NoActionBar">
|
||||
<!-- Customize your theme here. -->
|
||||
<item name="colorPrimary">@color/colorPrimary</item>
|
||||
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
|
||||
<item name="colorAccent">@color/colorAccent</item>
|
||||
</style>
|
||||
|
||||
</resources>
|
||||
@@ -1,7 +1,7 @@
|
||||
# Positions
|
||||
# RegExpUtils
|
||||
|
||||
#### 介绍
|
||||
安卓位置应用,有关于地理位置的相关应用。
|
||||
正则表达式工具集。
|
||||
|
||||
#### 软件架构
|
||||
适配安卓应用 [AIDE Pro] 的 Gradle 编译结构。
|
||||
@@ -10,7 +10,7 @@
|
||||
|
||||
#### Gradle 编译说明
|
||||
调试版编译命令 :gradle assembleBetaDebug
|
||||
阶段版编译命令 :bash .winboll/bashPublishAPKAddTag.sh positions
|
||||
阶段版编译命令 :bash .winboll/bashPublishAPKAddTag.sh regexputils
|
||||
|
||||
#### 使用说明
|
||||
|
||||
@@ -18,18 +18,19 @@ def genVersionName(def versionName){
|
||||
}
|
||||
|
||||
android {
|
||||
|
||||
compileSdkVersion 32
|
||||
buildToolsVersion "32.0.0"
|
||||
|
||||
defaultConfig {
|
||||
applicationId "cc.winboll.studio.positions"
|
||||
applicationId "cc.winboll.studio.regexputils"
|
||||
minSdkVersion 24
|
||||
targetSdkVersion 30
|
||||
versionCode 1
|
||||
// versionName 更新后需要手动设置
|
||||
// .winboll/winbollBuildProps.properties 文件的 stageCount=0
|
||||
// 项目模块目录的 build.gradle 文件的 stageCount=0
|
||||
// Gradle编译环境下合起来的 versionName 就是 "${versionName}.0"
|
||||
versionName "15.0"
|
||||
versionName "15.10"
|
||||
if(true) {
|
||||
versionName = genVersionName("${versionName}")
|
||||
}
|
||||
@@ -45,32 +46,4 @@ android {
|
||||
|
||||
dependencies {
|
||||
api fileTree(dir: 'libs', include: ['*.jar'])
|
||||
|
||||
// 谷歌定位服务核心依赖(FusedLocationProviderClient所在库)
|
||||
api 'com.google.android.gms:play-services-location:21.0.1'
|
||||
|
||||
// SSH
|
||||
api 'com.jcraft:jsch:0.1.55'
|
||||
// Html 解析
|
||||
api 'org.jsoup:jsoup:1.13.1'
|
||||
// 二维码类库
|
||||
api 'com.google.zxing:core:3.4.1'
|
||||
api 'com.journeyapps:zxing-android-embedded:3.6.0'
|
||||
// 应用介绍页类库
|
||||
api 'io.github.medyo:android-about-page:2.0.0'
|
||||
// 吐司类库
|
||||
api 'com.github.getActivity:ToastUtils:10.5'
|
||||
// 网络连接类库
|
||||
api 'com.squareup.okhttp3:okhttp:4.4.1'
|
||||
// AndroidX 类库
|
||||
api 'androidx.appcompat:appcompat:1.1.0'
|
||||
api 'com.google.android.material:material:1.4.0'
|
||||
//api 'androidx.viewpager:viewpager:1.0.0'
|
||||
//api 'androidx.vectordrawable:vectordrawable:1.1.0'
|
||||
//api 'androidx.vectordrawable:vectordrawable-animated:1.1.0'
|
||||
//api 'androidx.fragment:fragment:1.1.0'
|
||||
|
||||
api 'cc.winboll.studio:libaes:15.9.3'
|
||||
api 'cc.winboll.studio:libapputils:15.8.5'
|
||||
api 'cc.winboll.studio:libappbase:15.9.5'
|
||||
}
|
||||
8
regexputils/build.properties
Normal file
@@ -0,0 +1,8 @@
|
||||
#Created by .winboll/winboll_app_build.gradle
|
||||
#Mon Oct 06 20:51:16 HKT 2025
|
||||
stageCount=1
|
||||
libraryProject=
|
||||
baseVersion=15.10
|
||||
publishVersion=15.10.0
|
||||
buildCount=0
|
||||
baseBetaVersion=15.10.1
|
||||
17
regexputils/proguard-rules.pro
vendored
Normal file
@@ -0,0 +1,17 @@
|
||||
# Add project specific ProGuard rules here.
|
||||
# By default, the flags in this file are appended to flags specified
|
||||
# in C:\tools\adt-bundle-windows-x86_64-20131030\sdk/tools/proguard/proguard-android.txt
|
||||
# You can edit the include path and order by changing the proguardFiles
|
||||
# directive in build.gradle.
|
||||
#
|
||||
# For more details, see
|
||||
# http://developer.android.com/guide/developing/tools/proguard.html
|
||||
|
||||
# Add any project specific keep options here:
|
||||
|
||||
# If your project uses WebView with JS, uncomment the following
|
||||
# and specify the fully qualified class name to the JavaScript interface
|
||||
# class:
|
||||
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
|
||||
# public *;
|
||||
#}
|
||||
6
regexputils/src/beta/res/values-zh/strings.xml
Normal file
@@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
|
||||
<string name="app_name">正则工具☆</string>
|
||||
|
||||
</resources>
|
||||
@@ -1,6 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
|
||||
<string name="app_name">Positions +</string>
|
||||
<string name="app_name">RegExpUtils+</string>
|
||||
|
||||
</resources>
|
||||
39
regexputils/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1,39 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<manifest
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
package="cc.winboll.studio.regexputils">
|
||||
|
||||
<!-- 拥有完全的网络访问权限 -->
|
||||
<uses-permission android:name="android.permission.INTERNET"/>
|
||||
|
||||
<application
|
||||
android:name="cc.winboll.studio.regexputils.App"
|
||||
android:allowBackup="true"
|
||||
android:icon="@drawable/ic_regexputils"
|
||||
android:label="@string/app_name"
|
||||
android:theme="@style/AppTheme"
|
||||
android:resizeableActivity="true">
|
||||
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:label="@string/app_name">
|
||||
|
||||
<intent-filter>
|
||||
|
||||
<action android:name="android.intent.action.MAIN"/>
|
||||
|
||||
<category android:name="android.intent.category.LAUNCHER"/>
|
||||
|
||||
</intent-filter>
|
||||
|
||||
</activity>
|
||||
|
||||
<activity android:name="cc.winboll.studio.regexputils.develop.CrashHandler$CrashActiviy"/>
|
||||
|
||||
<meta-data
|
||||
android:name="android.max_aspect"
|
||||
android:value="4.0"/>
|
||||
|
||||
</application>
|
||||
|
||||
</manifest>
|
||||
@@ -0,0 +1,11 @@
|
||||
package cc.winboll.studio.regexputils;
|
||||
|
||||
import cc.winboll.studio.regexputils.develop.WinBollBase;
|
||||
|
||||
public class App extends WinBollBase {
|
||||
|
||||
public static final String TAG = "App";
|
||||
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,446 @@
|
||||
package cc.winboll.studio.regexputils;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.content.Context;
|
||||
import android.content.SharedPreferences;
|
||||
import android.graphics.Bitmap;
|
||||
import android.icu.text.SimpleDateFormat;
|
||||
import android.os.Bundle;
|
||||
import android.view.KeyEvent;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup.LayoutParams;
|
||||
import android.view.inputmethod.EditorInfo;
|
||||
import android.view.inputmethod.InputMethodManager;
|
||||
import android.webkit.WebView;
|
||||
import android.webkit.WebViewClient;
|
||||
import android.widget.AdapterView;
|
||||
import android.widget.ArrayAdapter;
|
||||
import android.widget.Button;
|
||||
import android.widget.EditText;
|
||||
import android.widget.LinearLayout;
|
||||
import android.widget.ListPopupWindow;
|
||||
import android.widget.TextView;
|
||||
import android.widget.Toast;
|
||||
import cc.winboll.studio.regexputils.develop.LogUtils;
|
||||
import cc.winboll.studio.regexputils.develop.LogView;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashSet;
|
||||
import java.util.Set;
|
||||
import java.util.regex.Matcher;
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
public class MainActivity extends Activity {
|
||||
|
||||
public static final String TAG = MainActivity.class.getSimpleName();
|
||||
|
||||
TextView mtvPattern;
|
||||
EditText metPattern;
|
||||
TextView mtvRewrite;
|
||||
EditText metRewrite;
|
||||
TextView mtvMatchText;
|
||||
EditText metMatchText;
|
||||
TextView mtvResult;
|
||||
TextView mtvOnlineHelp;
|
||||
Button mbtnFavorite;
|
||||
WebView mWebView;
|
||||
URLEditText mURLEditText;
|
||||
SharedPreferences mSharedPreferences;
|
||||
Set<String> mSetStringFavorite;
|
||||
String mszDefaultOnlineHelp;
|
||||
LogView mLogView;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_main);
|
||||
|
||||
mLogView = new LogView(this);
|
||||
LinearLayout mllLog = findViewById(R.id.activitymainLinearLayout1);
|
||||
mllLog.addView(mLogView);
|
||||
mLogView.startWatching();
|
||||
|
||||
initView();
|
||||
|
||||
LogUtils.d(TAG, "Created");
|
||||
|
||||
}
|
||||
|
||||
void initView() {
|
||||
//Toolbar toolbar= findViewById(R.id.toolbar);
|
||||
//setSupportActionBar(toolbar);
|
||||
|
||||
mSharedPreferences = getSharedPreferences("SP", Context.MODE_PRIVATE);
|
||||
|
||||
mtvPattern = findViewById(R.id.activitymainTextView2);
|
||||
metPattern = findViewById(R.id.activitymainEditText1);
|
||||
mtvRewrite = findViewById(R.id.activitymainTextView3);
|
||||
metRewrite = findViewById(R.id.activitymainEditText3);
|
||||
mtvMatchText = findViewById(R.id.activitymainTextView5);
|
||||
metMatchText = findViewById(R.id.activitymainEditText2);
|
||||
mtvResult = findViewById(R.id.activitymainTextView1);
|
||||
mWebView = findViewById(R.id.activitymainWebView1);
|
||||
mtvOnlineHelp = findViewById(R.id.activitymainTextView4);
|
||||
mURLEditText = findViewById(R.id.activitymainURLEditText1);
|
||||
mbtnFavorite = findViewById(R.id.activitymainButton1);
|
||||
|
||||
mtvOnlineHelp.setText(getString(R.string.tv_onlinehelp));
|
||||
//系统默认会通过手机浏览器打开网页,为了能够直接通过WebView显示网页,则必须设置
|
||||
mWebView.setWebViewClient(new WebViewClient(){
|
||||
@Override
|
||||
public boolean shouldOverrideUrlLoading(WebView view, String url) {
|
||||
//使用WebView加载显示url
|
||||
view.loadUrl(url);
|
||||
//返回true
|
||||
return true;
|
||||
}
|
||||
});
|
||||
|
||||
mSetStringFavorite = mSharedPreferences.getStringSet("mListFavorite", new HashSet<String>());
|
||||
String szURL = mSharedPreferences.getString("mWebView", getString(R.string.sz_defaultonlinehelp));
|
||||
addDefaultURLFavorite();
|
||||
mWebView.loadUrl(szURL);
|
||||
mWebView.setWebViewClient(new WebViewClient(){
|
||||
@Override
|
||||
public void onPageStarted(WebView view, String url, Bitmap favicon) {
|
||||
//设定加载开始的操作
|
||||
mURLEditText.setText(url);
|
||||
for (String sz : mSetStringFavorite) {
|
||||
if (sz.equals(url)) {
|
||||
//mURLEditText.setFavoriteYes();
|
||||
mbtnFavorite.setText("★");
|
||||
return;
|
||||
}
|
||||
}
|
||||
mbtnFavorite.setText("☆");
|
||||
//mURLEditText.setFavoriteNo();
|
||||
}
|
||||
|
||||
});
|
||||
mURLEditText.setText(szURL);
|
||||
mURLEditText.setOnEditorActionListener(
|
||||
new TextView.OnEditorActionListener() {
|
||||
@Override
|
||||
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
|
||||
if (actionId == EditorInfo.IME_ACTION_DONE) {
|
||||
onGoto(null);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});
|
||||
/* mURLEditText.setOnTouchListener(new View.OnTouchListener() {
|
||||
@Override
|
||||
public boolean onTouch(View view, MotionEvent event) {
|
||||
final int DRAWABLE_LEFT = 0;
|
||||
//final int DRAWABLE_TOP = 1;
|
||||
final int DRAWABLE_RIGHT = 2;
|
||||
//final int DRAWABLE_BOTTOM = 3;
|
||||
if (event.getAction() == MotionEvent.ACTION_UP) {
|
||||
if (event.getX() >= (mURLEditText.getWidth() - mURLEditText
|
||||
.getCompoundDrawables()[DRAWABLE_RIGHT].getBounds().width())) {
|
||||
|
||||
return true;
|
||||
} else if (event.getX() <= (mURLEditText
|
||||
.getCompoundDrawables()[DRAWABLE_LEFT].getBounds().width())) {
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});*/
|
||||
mtvPattern.setText(getString(R.string.tv_pattern));
|
||||
metPattern.setText(mSharedPreferences.getString("metPattern", "[^n]"));
|
||||
metPattern.setOnEditorActionListener(
|
||||
new TextView.OnEditorActionListener() {
|
||||
@Override
|
||||
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
|
||||
if (actionId == EditorInfo.IME_ACTION_DONE) {
|
||||
onRegExp(null);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});
|
||||
mtvRewrite.setText(getString(R.string.tv_rewrite));
|
||||
metRewrite.setText(mSharedPreferences.getString("metRewrite", ""));
|
||||
metRewrite.setOnEditorActionListener(
|
||||
new TextView.OnEditorActionListener() {
|
||||
@Override
|
||||
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
|
||||
if (actionId == EditorInfo.IME_ACTION_DONE) {
|
||||
onRegExpRewrite(null);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});
|
||||
mtvMatchText.setText(getString(R.string.tv_matchtext));
|
||||
metMatchText.setText(mSharedPreferences.getString("metMatchText", "Test string 123."));
|
||||
metMatchText.setOnEditorActionListener(
|
||||
new TextView.OnEditorActionListener() {
|
||||
@Override
|
||||
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
|
||||
if (actionId == EditorInfo.IME_ACTION_DONE) {
|
||||
onRegExp(null);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
}
|
||||
|
||||
public void onEnableFavorite(View v) {
|
||||
changeURLFavorite();
|
||||
}
|
||||
|
||||
public void onShowFavoriteList(View v) {
|
||||
showListPopulWindow();
|
||||
}
|
||||
|
||||
void showListPopulWindow() {
|
||||
final String[] list = (String[])mSetStringFavorite.toArray(new String[0]);
|
||||
final ListPopupWindow listPopupWindow;
|
||||
listPopupWindow = new ListPopupWindow(this);
|
||||
listPopupWindow.setWidth(LayoutParams.WRAP_CONTENT);
|
||||
listPopupWindow.setHeight(LayoutParams.WRAP_CONTENT);
|
||||
listPopupWindow.setAdapter(new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, list));//用android内置布局,或设计自己的样式
|
||||
//设置下拉列表基准控件
|
||||
listPopupWindow.setAnchorView(mURLEditText);
|
||||
listPopupWindow.setModal(true);
|
||||
// 透明度
|
||||
//listPopupWindow.setBackgroundDrawable(new ColorDrawable(0x00ffffff));
|
||||
listPopupWindow.setBackgroundDrawable(getDrawable(R.drawable.bg_shadow));
|
||||
|
||||
listPopupWindow.setOnItemClickListener(new AdapterView.OnItemClickListener() {//设置项点击监听
|
||||
@Override
|
||||
public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
|
||||
mURLEditText.setText(list[i]);//展示选择的内容
|
||||
setWebViewURL(list[i]);
|
||||
listPopupWindow.dismiss();//如果已经选择了,隐藏起来
|
||||
}
|
||||
});
|
||||
listPopupWindow.show();//下拉列表展示出来
|
||||
}
|
||||
|
||||
public void onRegExp(View v) {
|
||||
mtvResult.setText("");
|
||||
regExp();
|
||||
hideKeyboard(v);
|
||||
}
|
||||
|
||||
public void onRegExpRewrite(View v) {
|
||||
mtvResult.setText("");
|
||||
regExpRewrite();
|
||||
hideKeyboard(v);
|
||||
}
|
||||
|
||||
//隐藏虚拟键盘
|
||||
//
|
||||
public static void hideKeyboard(View v) {
|
||||
InputMethodManager imm = ( InputMethodManager ) v.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
|
||||
if (imm.isActive()) {
|
||||
imm.hideSoftInputFromWindow(v.getApplicationWindowToken() , 0);
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
// 获取当前时间格式化后的字符串
|
||||
//
|
||||
String getCurrentTimeString() {
|
||||
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:SS");
|
||||
return sdf.format(System.currentTimeMillis());
|
||||
}
|
||||
|
||||
ArrayList<String> regExp() {
|
||||
ArrayList<String> listResult = new ArrayList<String>();
|
||||
|
||||
StringBuilder sbPrint = new StringBuilder();
|
||||
sbPrint.append(getString(R.string.sz_patternresult));
|
||||
sbPrint.append(" : ");
|
||||
sbPrint.append(getCurrentTimeString());
|
||||
|
||||
// 输出当前累积的消息并清理 sbPrint
|
||||
mtvResult.append(sbPrint.toString());
|
||||
sbPrint = new StringBuilder("\n");
|
||||
|
||||
try {
|
||||
String szMatchText = metMatchText.getText().toString();
|
||||
String szPattern = metPattern.getText().toString();
|
||||
mSharedPreferences = getSharedPreferences("SP", Context.MODE_PRIVATE);
|
||||
mSharedPreferences.edit().putString("metPattern", metPattern.getText().toString()).commit();
|
||||
mSharedPreferences.edit().putString("metMatchText", metMatchText.getText().toString()).commit();
|
||||
listResult = buildRegExpList(szMatchText, szPattern);
|
||||
if (listResult.size() > 0) {
|
||||
sbPrint.append("\nRegExp Result : ");
|
||||
sbPrint.append(listResult.size());
|
||||
} else {
|
||||
sbPrint.append("\nRegExp Result : 0");
|
||||
}
|
||||
} catch (Exception e) {
|
||||
LogUtils.d(TAG, "Exception : " + e.getMessage());
|
||||
}
|
||||
|
||||
mtvResult.append(sbPrint.toString());
|
||||
|
||||
LogUtils.d(TAG, "RegExp Done.");
|
||||
return listResult;
|
||||
}
|
||||
|
||||
void regExpRewrite() {
|
||||
StringBuilder sbPrint = new StringBuilder("\n");
|
||||
try {
|
||||
String szMatchText = metMatchText.getText().toString();
|
||||
String szPattern = metPattern.getText().toString();
|
||||
String szRewrite = metRewrite.getText().toString();
|
||||
mSharedPreferences = getSharedPreferences("SP", Context.MODE_PRIVATE);
|
||||
mSharedPreferences.edit().putString("metPattern", metPattern.getText().toString()).commit();
|
||||
mSharedPreferences.edit().putString("metMatchText", metMatchText.getText().toString()).commit();
|
||||
mSharedPreferences.edit().putString("metRewrite", metRewrite.getText().toString()).commit();
|
||||
int nRewriteCount = regExpRewrite(szMatchText, szPattern, szRewrite);
|
||||
if (nRewriteCount > 0) {
|
||||
sbPrint.append("\nRewrite Result : ");
|
||||
sbPrint.append(nRewriteCount);
|
||||
} else {
|
||||
sbPrint.append("\nRewrite Result : 0");
|
||||
}
|
||||
} catch (Exception e) {
|
||||
LogUtils.d(TAG, "Exception : " + e.getMessage());
|
||||
}
|
||||
mtvResult.append(sbPrint);
|
||||
LogUtils.d(TAG, "RegExp Rewrite Done.");
|
||||
}
|
||||
|
||||
//
|
||||
// 生成语法模板匹配到的所有字符串数组
|
||||
//
|
||||
ArrayList<String> buildRegExpList(String szMatchText, String szPattern) {
|
||||
ArrayList<String> listResult = new ArrayList<String>();
|
||||
StringBuilder sbPrint = new StringBuilder("\n");
|
||||
|
||||
Pattern pattern = Pattern.compile(szPattern, Pattern.MULTILINE);
|
||||
Matcher matcher = pattern.matcher(szMatchText);
|
||||
|
||||
boolean isNull = true;
|
||||
int nStart = 0;
|
||||
while (matcher.find(nStart)) {
|
||||
isNull = false;
|
||||
int start = matcher.start();
|
||||
int end = matcher.end();
|
||||
String group = matcher.group();
|
||||
sbPrint.append("\n");
|
||||
sbPrint.append(getString(R.string.sz_start));
|
||||
sbPrint.append("(");
|
||||
sbPrint.append(Integer.toString(start));
|
||||
sbPrint.append(") ");
|
||||
sbPrint.append(getString(R.string.sz_end));
|
||||
sbPrint.append("(");
|
||||
sbPrint.append(Integer.toString(end));
|
||||
sbPrint.append(") >>> ");
|
||||
sbPrint.append(group);
|
||||
|
||||
listResult.add(group);
|
||||
nStart = matcher.end();
|
||||
}
|
||||
|
||||
mtvResult.append(sbPrint.toString());
|
||||
|
||||
return listResult;
|
||||
}
|
||||
|
||||
int regExpRewrite(String szMatchText, String szPattern, String szRewrite) {
|
||||
int nRewriteCount = 0;
|
||||
ArrayList<String> listResult = regExp();
|
||||
|
||||
StringBuilder sbPrint = new StringBuilder("\n\n\n\n\n####################\n");
|
||||
sbPrint.append(getString(R.string.sz_rewriteresult));
|
||||
sbPrint.append(" : ");
|
||||
sbPrint.append(getCurrentTimeString());
|
||||
|
||||
// 输出当前累积的消息并清理 sbPrint
|
||||
mtvResult.append(sbPrint.toString());
|
||||
sbPrint = new StringBuilder();
|
||||
|
||||
for (int i = 0; i < listResult.size(); i++) {
|
||||
Pattern pattern = Pattern.compile(szPattern, Pattern.MULTILINE);
|
||||
Matcher matcher = pattern.matcher(listResult.get(i));
|
||||
if (matcher.find()) {
|
||||
nRewriteCount++;
|
||||
sbPrint.append("\n Rewrite ");
|
||||
sbPrint.append(nRewriteCount);
|
||||
sbPrint.append(" : \n");
|
||||
sbPrint.append(matcher.replaceAll(szRewrite));
|
||||
sbPrint.append(" \n");
|
||||
}
|
||||
}
|
||||
|
||||
mtvResult.append(sbPrint.toString());
|
||||
return nRewriteCount;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
mLogView.stopWatching();
|
||||
|
||||
mSharedPreferences.edit().putString("metPattern", metPattern.getText().toString()).commit();
|
||||
mSharedPreferences.edit().putString("metMatchText", metMatchText.getText().toString()).commit();
|
||||
mSharedPreferences.edit().putStringSet("mListFavorite", mSetStringFavorite).commit();
|
||||
|
||||
super.onDestroy();
|
||||
}
|
||||
|
||||
void changeURLFavorite() {
|
||||
//String szURL = getString(R.string.webview_helpurl);
|
||||
//setWebViewURL(szURL);
|
||||
String url = mURLEditText.getText().toString();
|
||||
|
||||
// 转换收藏开关
|
||||
for (String sz : mSetStringFavorite) {
|
||||
if (sz.equals(url)) {
|
||||
// 保护默认URL
|
||||
if (!getString(R.string.sz_defaultonlinehelp).equals(url)) {
|
||||
mSetStringFavorite.remove(sz);
|
||||
//mURLEditText.setFavoriteNo();
|
||||
mbtnFavorite.setText("☆");
|
||||
mSharedPreferences.edit().putStringSet("mListFavorite", mSetStringFavorite).commit();
|
||||
return;
|
||||
} else {
|
||||
Toast.makeText(getApplication(), "!☆", Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
}
|
||||
}
|
||||
mSetStringFavorite.add(url);
|
||||
mbtnFavorite.setText("★");
|
||||
//mURLEditText.setFavoriteYes();
|
||||
mSharedPreferences.edit().putStringSet("mListFavorite", mSetStringFavorite).commit();
|
||||
}
|
||||
|
||||
void addDefaultURLFavorite() {
|
||||
//String szURL = getString(R.string.webview_helpurl);
|
||||
//setWebViewURL(szURL);
|
||||
String szDefault = getString(R.string.sz_defaultonlinehelp);
|
||||
|
||||
// 转换收藏开关
|
||||
for (String sz : mSetStringFavorite) {
|
||||
if (sz.equals(szDefault)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
mSetStringFavorite.add(szDefault);
|
||||
//mURLEditText.setFavoriteYes();
|
||||
mbtnFavorite.setText("★");
|
||||
mSharedPreferences.edit().putStringSet("mListFavorite", mSetStringFavorite).commit();
|
||||
}
|
||||
|
||||
public void onGoto(View v) {
|
||||
setWebViewURL(mURLEditText.getText().toString());
|
||||
}
|
||||
|
||||
void setWebViewURL(String szURL) {
|
||||
mSharedPreferences.edit().putString("mWebView", szURL).commit();
|
||||
mWebView.loadUrl(szURL);
|
||||
mURLEditText.setText(szURL);
|
||||
}
|
||||
|
||||
public void onCleanLog(View view) {
|
||||
LogUtils.cleanLog();
|
||||
LogUtils.d(TAG, "Log cleaned");
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,13 @@
|
||||
package cc.winboll.studio.regexputils;
|
||||
|
||||
import cc.winboll.studio.regexputils.develop.WinBollBase;
|
||||
|
||||
public class RegExpUtils extends WinBollBase {
|
||||
|
||||
public static final String TAG = RegExpUtils.class.getSimpleName();
|
||||
|
||||
@Override
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,83 @@
|
||||
package cc.winboll.studio.regexputils;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.os.Build;
|
||||
import android.util.AttributeSet;
|
||||
import android.widget.EditText;
|
||||
|
||||
public class URLEditText extends EditText {
|
||||
//Drawable drawable_r;
|
||||
//Drawable drawable_l_no;
|
||||
//Drawable drawable_l_yes;
|
||||
|
||||
public static final String TAG = URLEditText.class.getSimpleName();
|
||||
|
||||
/**
|
||||
* 在java代码里new的时候会用到
|
||||
* @param context
|
||||
*/
|
||||
public URLEditText(Context context) {
|
||||
super(context);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* 在xml布局文件中使用时自动调用
|
||||
* @param context
|
||||
*/
|
||||
public URLEditText(Context context, AttributeSet attrs) {
|
||||
super(context, attrs);
|
||||
/*TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.test);
|
||||
String text = ta.getString(R.styleable.test_text);
|
||||
setText(text + " ZhanGSKen.CN");
|
||||
ta.recycle();*/
|
||||
/*drawable_r = getResources().getDrawable(R.mipmap.dropdown);
|
||||
//drawable_n.setBounds(0, 0, drawable_n.getMinimumWidth(),drawable_n.getMinimumHeight()); //此为必须写的
|
||||
drawable_r.setBounds(0, 0, 80, 80); //此为必须写的
|
||||
|
||||
drawable_l_no = getResources().getDrawable(R.mipmap.favorite_no);
|
||||
//drawable_n.setBounds(0, 0, drawable_n.getMinimumWidth(),drawable_n.getMinimumHeight()); //此为必须写的
|
||||
drawable_l_no.setBounds(0, 0, 80, 80); //此为必须写的
|
||||
|
||||
drawable_l_yes = getResources().getDrawable(R.mipmap.favorite_yes);
|
||||
//drawable_n.setBounds(0, 0, drawable_n.getMinimumWidth(),drawable_n.getMinimumHeight()); //此为必须写的
|
||||
drawable_l_yes.setBounds(0, 0, 80, 80); //此为必须写的
|
||||
|
||||
setFavoriteNo();*/
|
||||
//setMinWidth(100);
|
||||
//setPadding(10,0,10,0);
|
||||
//setPaddingRelative(0,0,0,0);
|
||||
}
|
||||
|
||||
/**
|
||||
* 不会自动调用,如果有默认style时,在第二个构造函数中调用
|
||||
* @param context
|
||||
* @param attrs
|
||||
* @param defStyleAttr
|
||||
*/
|
||||
public URLEditText(Context context, AttributeSet attrs, int defStyleAttr) {
|
||||
super(context, attrs, defStyleAttr);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 只有在API版本>21时才会用到
|
||||
* 不会自动调用,如果有默认style时,在第二个构造函数中调用
|
||||
* @param context
|
||||
* @param attrs
|
||||
* @param defStyleAttr
|
||||
* @param defStyleRes
|
||||
*/
|
||||
//@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
|
||||
public URLEditText(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
|
||||
super(context, attrs, defStyleAttr, defStyleRes);
|
||||
}
|
||||
|
||||
/* public void setFavoriteNo() {
|
||||
setCompoundDrawables(drawable_l_no, null, drawable_r, null);
|
||||
}
|
||||
public void setFavoriteYes() {
|
||||
setCompoundDrawables(drawable_l_yes, null, drawable_r, null);
|
||||
}*/
|
||||
}
|
||||
@@ -0,0 +1,237 @@
|
||||
package cc.winboll.studio.regexputils.develop;
|
||||
|
||||
/*
|
||||
* 应用异常处理类
|
||||
* 源码提供:AIDE
|
||||
* 源码维护:ZhanGSKen@QQ.COM
|
||||
*/
|
||||
|
||||
import android.app.Activity;
|
||||
import android.app.Application;
|
||||
import android.content.ActivityNotFoundException;
|
||||
import android.content.ClipData;
|
||||
import android.content.ClipboardManager;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.PackageInfo;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.content.res.Resources;
|
||||
import android.os.Build;
|
||||
import android.os.Bundle;
|
||||
import android.text.TextUtils;
|
||||
import android.view.Menu;
|
||||
import android.view.MenuItem;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.HorizontalScrollView;
|
||||
import android.widget.ScrollView;
|
||||
import android.widget.TextView;
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.PrintWriter;
|
||||
import java.io.StringWriter;
|
||||
import java.lang.Thread.UncaughtExceptionHandler;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Date;
|
||||
|
||||
public final class CrashHandler
|
||||
{
|
||||
|
||||
public static final UncaughtExceptionHandler DEFAULT_UNCAUGHT_EXCEPTION_HANDLER = Thread.getDefaultUncaughtExceptionHandler();
|
||||
|
||||
public static void init(Application app)
|
||||
{
|
||||
init(app, null);
|
||||
}
|
||||
|
||||
public static void init(final Application app, final String crashDir)
|
||||
{
|
||||
Thread.setDefaultUncaughtExceptionHandler(new UncaughtExceptionHandler(){
|
||||
|
||||
@Override
|
||||
public void uncaughtException(Thread thread, Throwable throwable)
|
||||
{
|
||||
try
|
||||
{
|
||||
tryUncaughtException(thread, throwable);
|
||||
}
|
||||
catch (Throwable e)
|
||||
{
|
||||
e.printStackTrace();
|
||||
if (DEFAULT_UNCAUGHT_EXCEPTION_HANDLER != null)
|
||||
DEFAULT_UNCAUGHT_EXCEPTION_HANDLER.uncaughtException(thread, throwable);
|
||||
}
|
||||
}
|
||||
|
||||
private void tryUncaughtException(Thread thread, Throwable throwable)
|
||||
{
|
||||
final String time = new SimpleDateFormat("yyyy_MM_dd-HH_mm_ss").format(new Date());
|
||||
File crashFile = new File(TextUtils.isEmpty(crashDir) ? new File(app.getExternalFilesDir(null), "crash")
|
||||
: new File(crashDir), "crash_" + time + ".txt");
|
||||
|
||||
String versionName = "unknown";
|
||||
long versionCode = 0;
|
||||
try
|
||||
{
|
||||
PackageInfo packageInfo = app.getPackageManager().getPackageInfo(app.getPackageName(), 0);
|
||||
versionName = packageInfo.versionName;
|
||||
versionCode = Build.VERSION.SDK_INT >= 28 ? packageInfo.getLongVersionCode()
|
||||
: packageInfo.versionCode;
|
||||
}
|
||||
catch (PackageManager.NameNotFoundException ignored)
|
||||
{}
|
||||
|
||||
String fullStackTrace; {
|
||||
StringWriter sw = new StringWriter();
|
||||
PrintWriter pw = new PrintWriter(sw);
|
||||
throwable.printStackTrace(pw);
|
||||
fullStackTrace = sw.toString();
|
||||
pw.close();
|
||||
}
|
||||
|
||||
StringBuilder sb = new StringBuilder();
|
||||
sb.append("************* Crash Head ****************\n");
|
||||
sb.append("Time Of Crash : ").append(time).append("\n");
|
||||
sb.append("Device Manufacturer: ").append(Build.MANUFACTURER).append("\n");
|
||||
sb.append("Device Model : ").append(Build.MODEL).append("\n");
|
||||
sb.append("Android Version : ").append(Build.VERSION.RELEASE).append("\n");
|
||||
sb.append("Android SDK : ").append(Build.VERSION.SDK_INT).append("\n");
|
||||
sb.append("App VersionName : ").append(versionName).append("\n");
|
||||
sb.append("App VersionCode : ").append(versionCode).append("\n");
|
||||
sb.append("************* Crash Head ****************\n");
|
||||
sb.append("\n").append(fullStackTrace);
|
||||
|
||||
String errorLog = sb.toString();
|
||||
|
||||
try
|
||||
{
|
||||
writeFile(crashFile, errorLog);
|
||||
}
|
||||
catch (IOException ignored)
|
||||
{}
|
||||
|
||||
gotoCrashActiviy:
|
||||
{
|
||||
Intent intent = new Intent(app, CrashActiviy.class);
|
||||
intent.addFlags(
|
||||
Intent.FLAG_ACTIVITY_NEW_TASK
|
||||
| Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_CLEAR_TASK
|
||||
);
|
||||
intent.putExtra(CrashActiviy.EXTRA_CRASH_INFO, errorLog);
|
||||
try
|
||||
{
|
||||
app.startActivity(intent);
|
||||
android.os.Process.killProcess(android.os.Process.myPid());
|
||||
System.exit(0);
|
||||
}
|
||||
catch (ActivityNotFoundException e)
|
||||
{
|
||||
e.printStackTrace();
|
||||
if (DEFAULT_UNCAUGHT_EXCEPTION_HANDLER != null)
|
||||
DEFAULT_UNCAUGHT_EXCEPTION_HANDLER.uncaughtException(thread, throwable);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void writeFile(File file, String content) throws IOException
|
||||
{
|
||||
File parentFile = file.getParentFile();
|
||||
if (parentFile != null && !parentFile.exists())
|
||||
{
|
||||
parentFile.mkdirs();
|
||||
}
|
||||
file.createNewFile();
|
||||
FileOutputStream fos = new FileOutputStream(file);
|
||||
fos.write(content.getBytes());
|
||||
try
|
||||
{
|
||||
fos.close();
|
||||
}
|
||||
catch (IOException e)
|
||||
{}
|
||||
}
|
||||
|
||||
});
|
||||
}
|
||||
|
||||
public static final class CrashActiviy extends Activity implements MenuItem.OnMenuItemClickListener
|
||||
{
|
||||
|
||||
private static final String EXTRA_CRASH_INFO = "crashInfo";
|
||||
|
||||
private String mLog;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState)
|
||||
{
|
||||
super.onCreate(savedInstanceState);
|
||||
mLog = getIntent().getStringExtra(EXTRA_CRASH_INFO);
|
||||
setContentView:
|
||||
{
|
||||
ScrollView contentView = new ScrollView(this);
|
||||
contentView.setFillViewport(true);
|
||||
HorizontalScrollView hw = new HorizontalScrollView(this);
|
||||
TextView message = new TextView(this); {
|
||||
int padding = dp2px(16);
|
||||
message.setPadding(padding, padding, padding, padding);
|
||||
message.setText(mLog);
|
||||
message.setTextIsSelectable(true);
|
||||
}
|
||||
hw.addView(message);
|
||||
contentView.addView(hw, ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
|
||||
setContentView(contentView);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed()
|
||||
{
|
||||
restart();
|
||||
}
|
||||
|
||||
private void restart()
|
||||
{
|
||||
PackageManager pm = getPackageManager();
|
||||
Intent intent = pm.getLaunchIntentForPackage(getPackageName());
|
||||
if (intent != null)
|
||||
{
|
||||
intent.addFlags(
|
||||
Intent.FLAG_ACTIVITY_NEW_TASK
|
||||
| Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_CLEAR_TASK
|
||||
);
|
||||
startActivity(intent);
|
||||
}
|
||||
finish();
|
||||
android.os.Process.killProcess(android.os.Process.myPid());
|
||||
System.exit(0);
|
||||
}
|
||||
|
||||
private int dp2px(final float dpValue)
|
||||
{
|
||||
final float scale = Resources.getSystem().getDisplayMetrics().density;
|
||||
return (int) (dpValue * scale + 0.5f);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onMenuItemClick(MenuItem item)
|
||||
{
|
||||
switch (item.getItemId())
|
||||
{
|
||||
case android.R.id.copy:
|
||||
ClipboardManager cm = (ClipboardManager) getSystemService(Context.CLIPBOARD_SERVICE);
|
||||
cm.setPrimaryClip(ClipData.newPlainText(getPackageName(), mLog));
|
||||
break;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu)
|
||||
{
|
||||
menu.add(0, android.R.id.copy, 0, android.R.string.copy).setOnMenuItemClickListener(this)
|
||||
.setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,95 @@
|
||||
package cc.winboll.studio.regexputils.develop;
|
||||
|
||||
/*
|
||||
* 应用日志监听类
|
||||
* 源码提供:https://blog.csdn.net/wuxueshuan/article/details/121852698?app_version=5.15.2&code=app_1562916241&csdn_share_tail=%7B%22type%22%3A%22blog%22%2C%22rType%22%3A%22article%22%2C%22rId%22%3A%22121852698%22%2C%22source%22%3A%22weixin_38986226%22%7D&uLinkId=usr1mkqgl919blen&utm_source=app
|
||||
* 源码维护:ZhanGSKen@QQ.COM
|
||||
*/
|
||||
|
||||
import android.os.FileObserver;
|
||||
import android.util.Log;
|
||||
|
||||
public class LogListener extends FileObserver {
|
||||
public final static String TAG = "LogListener";
|
||||
|
||||
public EventCallback callback;
|
||||
|
||||
public String mLogPath;
|
||||
|
||||
public LogListener(String path) {
|
||||
super(path);
|
||||
//Log.d(TAG, "LogListener(String path) path : " + path);
|
||||
mLogPath = path;
|
||||
}
|
||||
public void setEventCallback(EventCallback callback) {
|
||||
this.callback = callback;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEvent(int event, String path) {
|
||||
if (null != path && !"".equals(path)) {
|
||||
//String substring = path.substring(path.lastIndexOf(".") + 1);
|
||||
//Log.d(TAG, "path : " + path);
|
||||
//Log.d(TAG, "substring : " + substring);
|
||||
//Log.d(TAG, "event : " + event);
|
||||
|
||||
} else {
|
||||
return;
|
||||
}
|
||||
|
||||
int e = event & FileObserver.ALL_EVENTS;
|
||||
//Log.d(TAG, "event->e:" + e);
|
||||
switch (e) {
|
||||
case FileObserver.ACCESS:
|
||||
//Log.d(TAG, "文件操作___" + e + "__1打开文件后读取文件的操作");
|
||||
break;
|
||||
case FileObserver.MODIFY:
|
||||
//Log.d(TAG, "文件操作___" + e + "__2文件被修改");
|
||||
break;
|
||||
case FileObserver.ATTRIB:
|
||||
//Log.d(TAG, "文件操作___" + e + "__4属性变化");
|
||||
break;
|
||||
case FileObserver.CLOSE_WRITE:
|
||||
//Log.d(TAG, "文件操作___" + e + "__8文件写入或编辑后关闭");
|
||||
callback.onEvent(path);
|
||||
break;
|
||||
case FileObserver.CLOSE_NOWRITE:
|
||||
//录音时,最后一个有效回调是这个
|
||||
//Log.d(TAG, "文件操作___" + e + "__16只读文件被关闭");
|
||||
|
||||
//callback.onEvent(path);
|
||||
|
||||
|
||||
break;
|
||||
case FileObserver.OPEN:
|
||||
//Log.d(TAG, "文件操作___" + e + "__32文件被打开");
|
||||
break;
|
||||
case FileObserver.MOVED_FROM:
|
||||
//Log.d(TAG, "文件操作___" + e + "__64移出事件");//试了重命名先MOVED_FROM再MOVED_TO
|
||||
break;
|
||||
case FileObserver.MOVED_TO:
|
||||
//Log.d(TAG, "文件操作___" + e + "__128移入事件");
|
||||
break;
|
||||
case FileObserver.CREATE:
|
||||
//Log.d(TAG, "文件操作___" + e + "__256新建文件");//把文件移动给自己先CREATE在DELETE
|
||||
break;
|
||||
case FileObserver.DELETE:
|
||||
//Log.d(TAG, "文件操作___" + e + "__512有删除文件");//把文件移出去DELETE
|
||||
break;
|
||||
case FileObserver.DELETE_SELF:
|
||||
//Log.d(TAG, "文件操作___" + e + "__1024监听的这个文件夹被删除");
|
||||
break;
|
||||
case FileObserver.MOVE_SELF:
|
||||
//Log.d(TAG, "文件操作___" + e + "__2048监听的这个文件夹被移走");
|
||||
break;
|
||||
case FileObserver.ALL_EVENTS:
|
||||
//Log.d(TAG, "文件操作___" + e + "__4095全部操作");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
public interface EventCallback {
|
||||
void onEvent(String path);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,74 @@
|
||||
package cc.winboll.studio.regexputils.develop;
|
||||
|
||||
/*
|
||||
* 应用日志类
|
||||
* 源码维护:ZhanGSKen@QQ.COM
|
||||
*/
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.BufferedWriter;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStreamReader;
|
||||
import java.io.OutputStreamWriter;
|
||||
import java.text.SimpleDateFormat;
|
||||
|
||||
public class LogUtils {
|
||||
|
||||
public static final String TAG = "BaseApplication";
|
||||
|
||||
static SimpleDateFormat mSimpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:SS");
|
||||
|
||||
public static void d(String szTAG, String szMessage) {
|
||||
saveLog(szTAG, szMessage);
|
||||
}
|
||||
|
||||
static void saveLog(String szTAG, String szMessage) {
|
||||
try {
|
||||
File fLog = new File(WinBollBase._mszLogFilePath);
|
||||
//FileWriter fw = new FileWriter(fLog, Charset.defaultCharset(), true);
|
||||
//fw.append(mSimpleDateFormat.format(System.currentTimeMillis()) + "[" + szTAG + "]: " + szMessage + "\n");
|
||||
//fw.close();
|
||||
BufferedWriter out = null;
|
||||
out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(fLog, true), "UTF-8"));
|
||||
out.write(mSimpleDateFormat.format(System.currentTimeMillis()) + "[" + szTAG + "]: " + szMessage + "\n");
|
||||
out.close();
|
||||
|
||||
} catch (IOException e) {
|
||||
LogUtils.d(TAG, "IOException : " + e.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
public static String loadLog() {
|
||||
File fLog = new File(WinBollBase._mszLogFilePath);
|
||||
StringBuffer sb = new StringBuffer();
|
||||
try {
|
||||
//FileInputStream fileInputStream = new FileInputStream(fLog);
|
||||
//int size = fileInputStream.available();
|
||||
//for (int i = 0; i < size; i++) {
|
||||
// sb.append((char) fileInputStream.read());
|
||||
//}
|
||||
BufferedReader in = null;
|
||||
in = new BufferedReader(new InputStreamReader(new FileInputStream(fLog), "UTF-8"));
|
||||
String line = "";
|
||||
while ((line = in.readLine()) != null) {
|
||||
sb.append(line);
|
||||
sb.append("\n");
|
||||
}
|
||||
|
||||
} catch (IOException e) {
|
||||
LogUtils.d(TAG, "IOException : " + e.getMessage());
|
||||
}
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
public static void cleanLog() {
|
||||
File fLog = new File(WinBollBase._mszLogFilePath);
|
||||
if (fLog.exists()) {
|
||||
fLog.delete();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,91 @@
|
||||
package cc.winboll.studio.regexputils.develop;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.Color;
|
||||
import android.os.Handler;
|
||||
import android.os.Message;
|
||||
import android.util.AttributeSet;
|
||||
import android.widget.LinearLayout;
|
||||
import android.widget.ScrollView;
|
||||
import android.widget.TextView;
|
||||
import cc.winboll.studio.regexputils.develop.LogUtils;
|
||||
import java.lang.ref.WeakReference;
|
||||
|
||||
public class LogView extends LinearLayout {
|
||||
|
||||
public static final String TAG = "LogView";
|
||||
|
||||
final static int MSG_SHOW_LOG = 0;
|
||||
|
||||
LogListener mLogListener;
|
||||
ScrollView mScrollView;
|
||||
TextView mTextView;
|
||||
MyHandler mMyHandler;
|
||||
|
||||
public LogView(Context context) {
|
||||
super(context);
|
||||
initView(context);
|
||||
}
|
||||
|
||||
public LogView(Context context, AttributeSet attrs) {
|
||||
super(context, attrs);
|
||||
initView(context);
|
||||
}
|
||||
|
||||
void initView(Context context) {
|
||||
mScrollView = new ScrollView(context);
|
||||
mTextView = new TextView(context);
|
||||
mTextView.setTextColor(Color.GREEN);
|
||||
mTextView.setTextIsSelectable(true);
|
||||
|
||||
mScrollView.addView(mTextView);
|
||||
addView(mScrollView);
|
||||
|
||||
mMyHandler = new MyHandler(this);
|
||||
|
||||
mLogListener = new LogListener(WinBollBase._mszLogFolderPath);
|
||||
mLogListener.setEventCallback(new LogListener.EventCallback(){
|
||||
@Override
|
||||
public void onEvent(String path) {
|
||||
Message message = mMyHandler.obtainMessage(MSG_SHOW_LOG);
|
||||
mMyHandler.sendMessage(message);
|
||||
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public void startWatching() {
|
||||
mLogListener.startWatching();
|
||||
}
|
||||
|
||||
public void stopWatching() {
|
||||
mLogListener.stopWatching();
|
||||
}
|
||||
|
||||
static class MyHandler extends Handler {
|
||||
WeakReference<LogView> rv;
|
||||
MyHandler(LogView view) {
|
||||
rv = new WeakReference<LogView>(view);
|
||||
}
|
||||
public void handleMessage(Message msg) {
|
||||
final LogView view = rv.get();
|
||||
switch (msg.what) {
|
||||
case MSG_SHOW_LOG:{
|
||||
view.mTextView.setText(LogUtils.loadLog());
|
||||
view.mScrollView.post(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
view.mScrollView.fullScroll(ScrollView.FOCUS_DOWN);
|
||||
}
|
||||
});
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
super.handleMessage(msg);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,31 @@
|
||||
package cc.winboll.studio.regexputils.develop;
|
||||
|
||||
/*
|
||||
* WinBollBase
|
||||
* WINBOLL 安卓应用基础类
|
||||
* 源码提供:AIDE
|
||||
* 源码维护:ZhanGSKen@QQ.COM
|
||||
*/
|
||||
|
||||
import android.app.Application;
|
||||
|
||||
public class WinBollBase extends Application {
|
||||
|
||||
public static String _mszLogFolderPath;
|
||||
public static String _mszLogFilePath;
|
||||
public static String _mszLogFileName = "WinBollBase.log";
|
||||
public static String _mszHtmlFolderPath;
|
||||
public static String _mszHtmlFilePath;
|
||||
public static String _mszHtmlFileName = "BaseWebViewBody.txt";
|
||||
|
||||
@Override
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
CrashHandler.init(this);
|
||||
_mszLogFolderPath = getExternalFilesDir("logs").toString();
|
||||
_mszLogFilePath = _mszLogFolderPath + "/" + _mszLogFileName;
|
||||
_mszHtmlFolderPath = getExternalFilesDir("webviews").toString();
|
||||
_mszHtmlFilePath = _mszHtmlFolderPath + "/" + _mszHtmlFileName;
|
||||
}
|
||||
|
||||
}
|
||||
41
regexputils/src/main/res/drawable/bg_shadow.xml
Normal file
@@ -0,0 +1,41 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<layer-list xmlns:android="http://schemas.android.com/apk/res/android" >
|
||||
<!-- 阴影部分 -->
|
||||
<!-- 个人觉得更形象的表达:top代表下边的阴影高度,left代表右边的阴影宽度。其实也就是相对应的offset,solid中的颜色是阴影的颜色,也可以设置角度等等 -->
|
||||
<item
|
||||
android:left="2dp"
|
||||
android:top="2dp"
|
||||
android:right="2dp"
|
||||
android:bottom="2dp">
|
||||
<shape android:shape="rectangle" >
|
||||
<gradient
|
||||
android:angle="270"
|
||||
android:endColor="#0F000000"
|
||||
android:startColor="#0F000000" />
|
||||
<corners
|
||||
android:bottomLeftRadius="6dip"
|
||||
android:bottomRightRadius="6dip"
|
||||
android:topLeftRadius="6dip"
|
||||
android:topRightRadius="6dip" />
|
||||
</shape>
|
||||
</item>
|
||||
<!-- 背景部分 -->
|
||||
<!-- 形象的表达:bottom代表背景部分在上边缘超出阴影的高度,right代表背景部分在左边超出阴影的宽度(相对应的offset) -->
|
||||
<item
|
||||
android:left="3dp"
|
||||
android:top="3dp"
|
||||
android:right="3dp"
|
||||
android:bottom="5dp">
|
||||
<shape android:shape="rectangle" >
|
||||
<gradient
|
||||
android:angle="270"
|
||||
android:endColor="#FFFFFF"
|
||||
android:startColor="#FFFFFF" />
|
||||
<corners
|
||||
android:bottomLeftRadius="6dip"
|
||||
android:bottomRightRadius="6dip"
|
||||
android:topLeftRadius="6dip"
|
||||
android:topRightRadius="6dip" />
|
||||
</shape>
|
||||
</item>
|
||||
</layer-list>
|
||||
@@ -6,5 +6,5 @@
|
||||
android:top="15dp"
|
||||
android:right="15dp"
|
||||
android:bottom="15dp"
|
||||
android:drawable="@drawable/ic_positions"/>
|
||||
android:drawable="@drawable/ic_regexputils"/>
|
||||
</layer-list>
|
||||
@@ -7,5 +7,5 @@
|
||||
android:top="15dp"
|
||||
android:right="15dp"
|
||||
android:bottom="15dp"
|
||||
android:drawable="@drawable/ic_positions"/>
|
||||
android:drawable="@drawable/ic_regexputils"/>
|
||||
</layer-list>
|
||||
BIN
regexputils/src/main/res/drawable/ic_regexputils.png
Normal file
|
After Width: | Height: | Size: 2.3 KiB |
200
regexputils/src/main/res/layout/activity_main.xml
Normal file
@@ -0,0 +1,200 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<LinearLayout
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:orientation="vertical"
|
||||
android:padding="10dp">
|
||||
|
||||
<LinearLayout
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:orientation="vertical"
|
||||
android:focusable="true"
|
||||
android:focusableInTouchMode="true">
|
||||
|
||||
<TextView
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:id="@+id/activitymainTextView2"
|
||||
style="@style/columnTitleStyle"/>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="horizontal"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:gravity="center_vertical">
|
||||
|
||||
<EditText
|
||||
android:layout_width="0dp"
|
||||
android:ems="10"
|
||||
android:layout_height="wrap_content"
|
||||
android:id="@+id/activitymainEditText1"
|
||||
android:layout_weight="1.0"
|
||||
android:singleLine="true"
|
||||
android:imeOptions="actionDone"
|
||||
style="@style/contentEditBoxStyle"/>
|
||||
|
||||
<Button
|
||||
android:layout_width="40dp"
|
||||
android:layout_height="40dp"
|
||||
android:text="!"
|
||||
android:onClick="onRegExp"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<TextView
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:id="@+id/activitymainTextView3"
|
||||
style="@style/columnTitleStyle"/>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="horizontal"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content">
|
||||
|
||||
<EditText
|
||||
android:layout_width="0dp"
|
||||
android:ems="10"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_weight="1.0"
|
||||
android:singleLine="true"
|
||||
android:imeOptions="actionDone"
|
||||
android:id="@+id/activitymainEditText3"
|
||||
style="@style/contentEditBoxStyle"/>
|
||||
|
||||
<Button
|
||||
android:layout_width="40dp"
|
||||
android:layout_height="40dp"
|
||||
android:text="! $"
|
||||
android:onClick="onRegExpRewrite"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<TextView
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:id="@+id/activitymainTextView5"
|
||||
style="@style/columnTitleStyle"/>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="horizontal"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content">
|
||||
|
||||
<EditText
|
||||
android:layout_width="match_parent"
|
||||
android:ems="10"
|
||||
android:layout_height="64dp"
|
||||
android:singleLine="false"
|
||||
android:imeOptions="actionDone"
|
||||
android:id="@+id/activitymainEditText2"
|
||||
style="@style/contentEditBoxStyle"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<ScrollView
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="150dp"
|
||||
android:background="#FFDBDBDB">
|
||||
|
||||
<TextView
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:textIsSelectable="true"
|
||||
android:id="@+id/activitymainTextView1"
|
||||
style="@style/columnTitleStyle"/>
|
||||
|
||||
</ScrollView>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="horizontal"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="65dp"
|
||||
android:gravity="center_vertical">
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="vertical"
|
||||
android:layout_width="0dp"
|
||||
android:layout_height="match_parent"
|
||||
android:layout_weight="1.0"
|
||||
android:background="#FF000000"
|
||||
android:id="@+id/activitymainLinearLayout1">
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<Button
|
||||
android:layout_width="40dp"
|
||||
android:layout_height="40dp"
|
||||
android:text="↻"
|
||||
android:onClick="onCleanLog"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<ScrollView
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="0dp"
|
||||
android:layout_weight="1.0">
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="vertical"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content">
|
||||
|
||||
<TextView
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:id="@+id/activitymainTextView4"
|
||||
style="@style/columnTitleStyle"/>
|
||||
|
||||
<LinearLayout
|
||||
android:orientation="horizontal"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content">
|
||||
|
||||
<Button
|
||||
android:layout_width="40dp"
|
||||
android:layout_height="40dp"
|
||||
android:text="☆"
|
||||
android:id="@+id/activitymainButton1"
|
||||
android:onClick="onEnableFavorite"/>
|
||||
|
||||
<cc.winboll.studio.regexputils.URLEditText
|
||||
android:layout_width="0dp"
|
||||
android:ems="10"
|
||||
android:layout_height="wrap_content"
|
||||
android:singleLine="true"
|
||||
android:imeOptions="actionDone"
|
||||
android:id="@+id/activitymainURLEditText1"
|
||||
android:layout_weight="1.0"/>
|
||||
|
||||
<Button
|
||||
android:layout_width="40dp"
|
||||
android:layout_height="40dp"
|
||||
android:text="▼"
|
||||
android:onClick="onShowFavoriteList"/>
|
||||
|
||||
<Button
|
||||
android:layout_width="40dp"
|
||||
android:layout_height="40dp"
|
||||
android:text="→"
|
||||
android:onClick="onGoto"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
<WebView
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:id="@+id/activitymainWebView1"/>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
</ScrollView>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
9
regexputils/src/main/res/values-v21/styles.xml
Normal file
@@ -0,0 +1,9 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<style name="AppTheme" parent="@android:style/Theme.Material.Light.DarkActionBar">
|
||||
<item name="android:colorPrimary">@color/colorPrimary</item>
|
||||
<item name="android:colorPrimaryDark">@color/colorPrimaryDark</item>
|
||||
<item name="android:colorAccent">@color/colorAccent</item>
|
||||
<item name="android:navigationBarColor">?android:colorPrimary</item>
|
||||
</style>
|
||||
</resources>
|
||||
14
regexputils/src/main/res/values-zh/strings.xml
Normal file
@@ -0,0 +1,14 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<string name="app_name">正则工具</string>
|
||||
<string name="tv_pattern">语法模板</string>
|
||||
<string name="tv_rewrite">替换模板</string>
|
||||
<string name="tv_matchtext">匹配文本</string>
|
||||
<string name="tv_onlinehelp">联机帮助</string>
|
||||
<string name="sz_patternresult">匹配结果</string>
|
||||
<string name="sz_rewriteresult">替换结果</string>
|
||||
<string name="sz_isnull">为空。</string>
|
||||
<string name="sz_nomatcherfind">没有匹配的字符。</string>
|
||||
<string name="sz_start">开始</string>
|
||||
<string name="sz_end">结束</string>
|
||||
</resources>
|
||||
10
regexputils/src/main/res/values/colors.xml
Normal file
@@ -0,0 +1,10 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<color name="colorPrimary">#3F51B5</color>
|
||||
<color name="colorPrimaryDark">#303F9F</color>
|
||||
<color name="colorAccent">#FF5771FF</color>
|
||||
<color name="color_2C2C2C">#2C2C2C</color>
|
||||
<color name="color_4C4E55">#4C4E55</color>
|
||||
<color name="black">#FF000000</color>
|
||||
<color name="white">#FFFFFFFF</color>
|
||||
</resources>
|
||||
16
regexputils/src/main/res/values/strings.xml
Normal file
@@ -0,0 +1,16 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<string name="app_name">RegExpUtils</string>
|
||||
<string name="tv_pattern">Pattern</string>
|
||||
<string name="tv_rewrite">Rewrite</string>
|
||||
<string name="tv_matchtext">MatchText</string>
|
||||
<string name="tv_onlinehelp">OnlineHelp</string>
|
||||
<string name="sz_patternresult">PatternResult</string>
|
||||
<string name="sz_rewriteresult">RewriteResult</string>
|
||||
<string name="sz_isnull">is null.</string>
|
||||
<string name="sz_nomatcherfind">No matcher find.</string>
|
||||
<string name="sz_start">Start</string>
|
||||
<string name="sz_end">End</string>
|
||||
<string name="sz_Source">https://toscode.gitee.com/zhangsken/RegExpUtils</string>
|
||||
<string name="sz_defaultonlinehelp">https://tool.oschina.net/uploads/apidocs/jquery/regexp.html</string>
|
||||
</resources>
|
||||
41
regexputils/src/main/res/values/styles.xml
Normal file
@@ -0,0 +1,41 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<!-- Base application theme. -->
|
||||
<style name="AppTheme" parent="android:Theme.Material.Light.DarkActionBar">
|
||||
<!-- Customize your theme here. -->
|
||||
<item name="android:colorPrimary">@color/colorPrimary</item>
|
||||
<item name="android:colorPrimaryDark">@color/colorPrimaryDark</item>
|
||||
<item name="android:colorAccent">@color/colorAccent</item>
|
||||
</style>
|
||||
<!--popMenu的Style-->
|
||||
<style name="customPopMenuStyle" >
|
||||
<item name="android:itemBackground">@color/color_2C2C2C</item>
|
||||
<item name="android:dropDownListViewStyle">@style/popmenuDivier</item>
|
||||
<item name="android:textAppearanceSmallPopupMenu">@style/popmeuText</item>
|
||||
<item name="android:textAppearanceLargePopupMenu">@style/popmeuText</item>
|
||||
</style>
|
||||
<!-- popMenu的背景色-->
|
||||
<style name="popmenuStyle" >
|
||||
</style>
|
||||
<!--popmenu的字体颜色-->
|
||||
<style name="popmeuText">
|
||||
<item name="android:textColor">@color/white</item>
|
||||
<item name="android:textSize">14sp</item>
|
||||
<item name="android:gravity">center</item>
|
||||
</style>
|
||||
<!--popMenu分割线的颜色-->
|
||||
<style name="popmenuDivier">
|
||||
<item name="android:divider">@color/color_4C4E55</item>
|
||||
<item name="android:dividerHeight">1px</item>
|
||||
</style>
|
||||
<!--分栏标题的风格-->
|
||||
<style name="columnTitleStyle" >
|
||||
<item name="android:textSize">12sp</item>
|
||||
<item name="android:textColor">@color/colorAccent</item>
|
||||
<item name="android:gravity">center_vertical</item>
|
||||
</style>
|
||||
<!--内容编辑框的风格-->
|
||||
<style name="contentEditBoxStyle" >
|
||||
<item name="android:textSize">14sp</item>
|
||||
</style>
|
||||
</resources>
|
||||
@@ -66,6 +66,6 @@
|
||||
//include ':webpagesources'
|
||||
//rootProject.name = "webpagesources"
|
||||
|
||||
// Positions 项目编译设置
|
||||
//include ':positions'
|
||||
//rootProject.name = "positions"
|
||||
// RegExpUtils 项目编译设置
|
||||
//include ':regexputils'
|
||||
//rootProject.name = "regexputils"
|
||||
|
||||