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" | ||||
|   | ||||