| @@ -25,12 +25,13 @@ | |||
| * 使用Android Studio生成签名包 | |||
| * 配置```appCopyright=```启动页版权文本, 不要加双引号 | |||
| * 配置```appVendor=```启动页厂商名称, 不要加双引号 | |||
| * 配置```appUpdateUrl=```apk下载更新地址, 不要加双引号 | |||
| * 生成的apk路径为 `/app/release/app-release.apk` | |||
| > 使用`gradle`脚本打签名的正式包 | |||
| * 执行 ```gradlew assembleRelease``` 将使用`gradle.properties`文件里的配置进行打包 | |||
| * 如果需要自定义配置(无需修改`gradle.properties`文件), 执行 ```gradlew assembleRelease -PappHomeUrl="首页地址" -PappName="App桌面快捷方式名称(只能使用字符串)" -PappIconKey="图标名称" -PappIconKey="启动页版权文本" -PappCopyright="启动页厂商名称"``` 将使用命令行里的配置进行打包 | |||
| * 如果需要自定义配置(无需修改`gradle.properties`文件), 执行 ```gradlew assembleRelease -PappHomeUrl="首页地址" -PappName="App桌面快捷方式名称(只能使用字符串)" -PappIconKey="图标名称" -PappIconKey="启动页版权文本" -PappCopyright="启动页厂商名称" -PappUpdateUrl="apk下载更新地址"``` 将使用命令行里的配置进行打包 | |||
| * 生成的apk路径为 `/app/build/outputs/apk/release/app-release.apk` | |||
| > 帮助脚本 | |||
| @@ -10,7 +10,7 @@ android { | |||
| minSdkVersion 24 | |||
| targetSdkVersion 30 | |||
| versionCode 1 | |||
| versionName "1.0" | |||
| versionName "1.0.0.1" | |||
| testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" | |||
| @@ -25,6 +25,9 @@ android { | |||
| // App图标 | |||
| buildConfigField "String", "APP_ICON", "\"${project.properties.appIconKey}\"" | |||
| // App更新下载地址 | |||
| buildConfigField "String", "APP_UPDATE_URL", "\"${project.properties.appUpdateUrl}\"" | |||
| // AndroidManifest.xml占位符 | |||
| manifestPlaceholders = [ | |||
| // App名称 | |||
| @@ -32,7 +35,9 @@ android { | |||
| // App图标 | |||
| APP_ICON: "@mipmap/ic_launcher_${project.properties.appIconKey}", | |||
| // App圆角图标 | |||
| APP_ROUND_ICON: "@mipmap/ic_launcher_${project.properties.appIconKey}_round" | |||
| APP_ROUND_ICON: "@mipmap/ic_launcher_${project.properties.appIconKey}_round", | |||
| // 高德key | |||
| AMAP_KEY: "${project.properties.amapKey}", | |||
| ] | |||
| } | |||
| @@ -56,6 +61,7 @@ android { | |||
| resValue("string", "app_copyright", "\"${project.properties.appCopyright}\"") | |||
| resValue("string", "app_vendor", "\"${project.properties.appVendor}\"") | |||
| resValue("string", "app_name2", "\"${project.properties.appName}\"") | |||
| resValue("string", "app_update_url", "\"${project.properties.appUpdateUrl}\"") | |||
| } | |||
| debug { | |||
| signingConfig signingConfigs.release | |||
| @@ -63,6 +69,7 @@ android { | |||
| resValue("string", "app_copyright", "\"${project.properties.appCopyright}\"") | |||
| resValue("string", "app_vendor", "\"${project.properties.appVendor}\"") | |||
| resValue("string", "app_name2", "\"${project.properties.appName}\"") | |||
| resValue("string", "app_update_url", "\"${project.properties.appUpdateUrl}\"") | |||
| } | |||
| } | |||
| compileOptions { | |||
| @@ -48,7 +48,7 @@ | |||
| android:theme="@style/Theme.Nsgk_rural_web"> | |||
| <meta-data | |||
| android:name="com.amap.api.v2.apikey" | |||
| android:value="7e6911ad842a6a6f492d27efdbd42f30"/> | |||
| android:value="${AMAP_KEY}"/> | |||
| <activity | |||
| android:name=".WelcomeActivity" | |||
| @@ -80,6 +80,20 @@ | |||
| </receiver> | |||
| --> | |||
| <service android:name="com.amap.api.location.APSService"></service> | |||
| <provider | |||
| android:name="androidx.core.content.FileProvider" | |||
| android:authorities="${applicationId}.fileprovider" | |||
| android:exported="false" | |||
| android:grantUriPermissions="true" | |||
| android:permission="android.permission.MANAGE_DOCUMENTS"> | |||
| <!-- <intent-filter> | |||
| <action android:name="android.content.action.DOCUMENTS_PROVIDER" /> | |||
| </intent-filter>--> | |||
| <meta-data | |||
| android:name="android.support.FILE_PROVIDER_PATHS" | |||
| android:resource="@xml/file_paths" /> | |||
| </provider> | |||
| </application> | |||
| </manifest> | |||
| @@ -10,6 +10,7 @@ import android.os.Bundle; | |||
| import android.os.Handler; | |||
| import android.util.Log; | |||
| import android.view.KeyEvent; | |||
| import android.view.View; | |||
| import android.view.ViewGroup; | |||
| import android.webkit.ConsoleMessage; | |||
| import android.webkit.GeolocationPermissions; | |||
| @@ -18,6 +19,7 @@ import android.webkit.WebHistoryItem; | |||
| import android.webkit.WebSettings; | |||
| import android.webkit.WebView; | |||
| import android.widget.LinearLayout; | |||
| import android.widget.ProgressBar; | |||
| import android.widget.RelativeLayout; | |||
| import android.widget.Toast; | |||
| @@ -54,6 +56,7 @@ public class FullscreenActivity extends AppCompatActivity { | |||
| private String m_lastUrl; | |||
| private String m_mainUrl; | |||
| private View m_progressIndicator; | |||
| @SuppressLint("SetJavaScriptEnabled") | |||
| @Override | |||
| @@ -167,6 +170,18 @@ public class FullscreenActivity extends AppCompatActivity { | |||
| mAgentWeb.getJsInterfaceHolder().addJavaObject("_Native_object", envWindowObject); | |||
| mAgentWeb.getJsInterfaceHolder().addJavaObject("Android", envWindowObject); | |||
| // 等待条指示器 | |||
| m_progressIndicator = getLayoutInflater().inflate(R.layout.location_indicator, null); | |||
| m_progressIndicator.findViewById(R.id.terminate_button).setOnClickListener((View view) -> { | |||
| envWindowObject.TerminateLocation(); | |||
| }); | |||
| m_progressIndicator.setVisibility(View.GONE); | |||
| m_progressIndicator.setZ(99); | |||
| RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT); | |||
| params.addRule(RelativeLayout.CENTER_IN_PARENT); | |||
| ((RelativeLayout)findViewById(R.id.ll)).addView(m_progressIndicator, params); | |||
| /* 上边url是各个APP项目的入口地址 | |||
| 事项审批 http://116.255.223.226:82/yinnongLogin 图标 ic_launcher_sxsp 或者 ic_launcher_yhzl | |||
| 阳光村务(村级事项) http://116.255.223.226:82/sunVillage_info/login 图标 ic_launcher_ygcw | |||
| @@ -215,16 +230,21 @@ public class FullscreenActivity extends AppCompatActivity { | |||
| @Override | |||
| public boolean onKeyDown(int keyCode, KeyEvent event) { | |||
| // if (keyCode == KeyEvent.KEYCODE_BACK && mWebView.canGoBack()) { | |||
| // mWebView.goBack(); | |||
| // return true; | |||
| // } | |||
| if (keyCode == KeyEvent.KEYCODE_BACK && HandleBackKeyEvent(1)) { | |||
| return true; | |||
| } | |||
| if (mAgentWeb.handleKeyEvent(keyCode, event)) { | |||
| return true; | |||
| } | |||
| return super.onKeyDown(keyCode, event); | |||
| } | |||
| @Override | |||
| public void onBackPressed() | |||
| { | |||
| if(!HandleBackKeyEvent(2)) | |||
| super.onBackPressed(); | |||
| } | |||
| @Override | |||
| protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { | |||
| @@ -250,19 +270,6 @@ public class FullscreenActivity extends AppCompatActivity { | |||
| } | |||
| } | |||
| @Override | |||
| public void onBackPressed() | |||
| { | |||
| WebView webView = GetWebView(); | |||
| if (null != webView && !webView.canGoBack() && !IsInDefaultPage()) { | |||
| webView.loadUrl(NSConstants.AppHomeUrl()); | |||
| webView.clearHistory(); | |||
| //History(); | |||
| return; | |||
| } | |||
| super.onBackPressed(); | |||
| } | |||
| private NSPreference GetPreference() | |||
| { | |||
| if(null == preference) | |||
| @@ -337,4 +344,38 @@ public class FullscreenActivity extends AppCompatActivity { | |||
| Log.d(ID_TAG, StrUtil.format("{}: {} {}", i, item.getTitle(), item.getUrl())); | |||
| } | |||
| } | |||
| public void SetIndicatorVisible(boolean on) | |||
| { | |||
| m_progressIndicator.setVisibility(on ? View.VISIBLE : View.GONE); | |||
| } | |||
| private boolean HandleBackKeyEvent(int from) | |||
| { | |||
| if(!IsRecordUrl()) | |||
| return false; | |||
| WebView webView = GetWebView(); | |||
| if(null == webView) | |||
| return false; | |||
| boolean inDefaultPage = IsInDefaultPage(); | |||
| //System.err.println(m_lastUrl +" -> " + inDefaultPage + " = " + webView.canGoBack()); | |||
| if(inDefaultPage) | |||
| { | |||
| if(from == 1 && !webView.canGoBack()) | |||
| { | |||
| finish(); | |||
| return true; | |||
| } | |||
| else | |||
| return false; | |||
| } | |||
| if (!webView.canGoBack() && !inDefaultPage) { | |||
| webView.loadUrl(NSConstants.AppHomeUrl()); | |||
| webView.clearHistory(); | |||
| //History(); | |||
| return true; | |||
| } | |||
| return false; | |||
| } | |||
| } | |||
| @@ -186,7 +186,7 @@ public class NSAMapLocation implements AMapLocationListener | |||
| //设置场景模式后最好调用一次stop,再调用start以保证场景模式生效 | |||
| mLocationClient.stopLocation(); | |||
| Log.i(ID_TAG, "定位停止"); | |||
| Log.i(ID_TAG, "高德定位停止"); | |||
| } | |||
| public void Shutdown() | |||
| @@ -197,6 +197,7 @@ public class NSAMapLocation implements AMapLocationListener | |||
| return; | |||
| } | |||
| TerminateRead(); | |||
| Stop(); | |||
| mLocationClient.onDestroy(); | |||
| @@ -205,10 +206,10 @@ public class NSAMapLocation implements AMapLocationListener | |||
| { | |||
| m_thread.quit(); | |||
| m_thread = null; | |||
| Log.d(ID_TAG, "内建线程结束"); | |||
| Log.d(ID_TAG, "高德定位内建线程结束"); | |||
| } | |||
| Log.i(ID_TAG, "定位销毁"); | |||
| Log.i(ID_TAG, "高德定位销毁"); | |||
| } | |||
| public void PreInit() | |||
| @@ -346,8 +347,7 @@ public class NSAMapLocation implements AMapLocationListener | |||
| @Override | |||
| public void onLocationChanged(AMapLocation amapLocation) | |||
| { | |||
| Log.d(ID_TAG, "onLocationChanged线程: " + Thread.currentThread().getId()); | |||
| System.err.println(amapLocation); | |||
| Log.d(ID_TAG, "AMap定位回调线程: " + Thread.currentThread().getId() + " -> " + amapLocation); | |||
| SetLastLocation(amapLocation); | |||
| if(null != amapLocation) | |||
| { | |||
| @@ -446,6 +446,15 @@ public class NSAMapLocation implements AMapLocationListener | |||
| return m_lastLocation; | |||
| } | |||
| public void TerminateRead() | |||
| { | |||
| synchronized(m_lock) | |||
| { | |||
| Log.i(ID_TAG, "AMap中止定位"); | |||
| m_lock.notifyAll(); | |||
| } | |||
| } | |||
| public void CleanLastLocation() | |||
| { | |||
| m_lastLocation = null; | |||
| @@ -0,0 +1,84 @@ | |||
| package com.nsgk.ruralWeb.location; | |||
| import android.annotation.SuppressLint; | |||
| import android.app.Activity; | |||
| import android.content.Context; | |||
| import android.location.Criteria; | |||
| import android.location.Location; | |||
| import android.location.LocationManager; | |||
| import android.os.Build; | |||
| import android.os.CancellationSignal; | |||
| import android.util.Log; | |||
| import android.widget.Toast; | |||
| import com.nsgk.ruralWeb.sys.NSConstants; | |||
| import java.util.HashMap; | |||
| import java.util.List; | |||
| import java.util.Map; | |||
| import java.util.concurrent.CompletableFuture; | |||
| import java.util.concurrent.TimeUnit; | |||
| public class NSLastKnownLocation | |||
| { | |||
| private static final String ID_TAG = NSLastKnownLocation.class.getName(); | |||
| private final Context m_context; | |||
| private NSLocationInfo m_lastLocation = null; | |||
| public NSLastKnownLocation(Context context) | |||
| { | |||
| m_context = context; | |||
| } | |||
| @SuppressLint("MissingPermission") | |||
| public NSLocationInfo Read(String type) | |||
| { | |||
| NSLocationInfo loc = null; | |||
| try | |||
| { | |||
| LocationManager lm = (LocationManager) m_context.getSystemService(Context.LOCATION_SERVICE); | |||
| List<String> allProviders = lm.getAllProviders(); | |||
| Log.i(ID_TAG, "使用最近所有定位提供器: " + allProviders); | |||
| Map<String, NSLocationInfo> map = new HashMap<>(); | |||
| for(String provider : allProviders) | |||
| { | |||
| Location lastKnownLocation = lm.getLastKnownLocation(provider); | |||
| if(null == lastKnownLocation) | |||
| continue; | |||
| NSLocationInfo l = new NSLocationInfo(provider, lastKnownLocation.getLongitude(), lastKnownLocation.getLatitude()); | |||
| map.put(provider, l); | |||
| } | |||
| Log.i(ID_TAG, "最近定位提供器结果: " + map); | |||
| if(null != type && !type.isEmpty() && map.containsKey(type)) | |||
| loc = map.get(type); | |||
| else if(map.containsKey(LocationManager.GPS_PROVIDER)) | |||
| loc = map.get(LocationManager.GPS_PROVIDER); | |||
| else if(map.containsKey(LocationManager.NETWORK_PROVIDER)) | |||
| loc = map.get(LocationManager.NETWORK_PROVIDER); | |||
| else if(map.containsKey(NSConstants.FUSED_PROVIDER)) | |||
| loc = map.get(NSConstants.FUSED_PROVIDER); | |||
| else if(map.containsKey(LocationManager.PASSIVE_PROVIDER)) | |||
| loc = map.get(LocationManager.PASSIVE_PROVIDER); | |||
| if(null == loc) | |||
| Log.i(ID_TAG, "所有提供器无最近定位"); | |||
| else | |||
| { | |||
| Log.i(ID_TAG, "使用提供器最近定位: " + loc.provider); | |||
| m_lastLocation = loc; | |||
| } | |||
| } | |||
| catch(Throwable e) | |||
| { | |||
| e.printStackTrace(); | |||
| } | |||
| return loc; | |||
| } | |||
| public NSLocationInfo GetLastLocation() | |||
| { | |||
| return m_lastLocation; | |||
| } | |||
| } | |||
| @@ -159,7 +159,7 @@ public class NSRealtimeLocation implements LocationListener | |||
| m_locationManager.removeUpdates(this); | |||
| m_provider = null; | |||
| Log.i(ID_TAG, "定位停止"); | |||
| Log.i(ID_TAG, "实时定位停止"); | |||
| } | |||
| public void Shutdown() | |||
| @@ -170,6 +170,7 @@ public class NSRealtimeLocation implements LocationListener | |||
| return; | |||
| } | |||
| TerminateRead(); | |||
| Stop(); | |||
| m_locationManager = null; | |||
| @@ -177,10 +178,10 @@ public class NSRealtimeLocation implements LocationListener | |||
| { | |||
| m_thread.quit(); | |||
| m_thread = null; | |||
| Log.d(ID_TAG, "内建线程结束"); | |||
| Log.d(ID_TAG, "实时定位内建线程结束"); | |||
| } | |||
| Log.i(ID_TAG, "定位销毁"); | |||
| Log.i(ID_TAG, "实时定位销毁"); | |||
| } | |||
| public boolean IsInitialized() | |||
| @@ -350,8 +351,7 @@ public class NSRealtimeLocation implements LocationListener | |||
| @Override | |||
| public void onLocationChanged(@NonNull Location location) | |||
| { | |||
| Log.d(ID_TAG, "onLocationChanged线程: " + Thread.currentThread().getId()); | |||
| System.err.println(location); | |||
| Log.d(ID_TAG, "系统实时定位线程: " + Thread.currentThread().getId() + " -> " + location); | |||
| SetLastLocation(location); | |||
| //if(null != location) | |||
| { | |||
| @@ -368,4 +368,13 @@ public class NSRealtimeLocation implements LocationListener | |||
| } | |||
| } | |||
| } | |||
| public void TerminateRead() | |||
| { | |||
| synchronized(m_lock) | |||
| { | |||
| Log.i(ID_TAG, "系统实时中止定位"); | |||
| m_lock.notifyAll(); | |||
| } | |||
| } | |||
| } | |||
| @@ -7,6 +7,7 @@ import android.location.Criteria; | |||
| import android.location.Location; | |||
| import android.location.LocationManager; | |||
| import android.os.Build; | |||
| import android.os.CancellationSignal; | |||
| import android.util.Log; | |||
| import android.widget.Toast; | |||
| @@ -14,92 +15,28 @@ import java.util.HashMap; | |||
| import java.util.List; | |||
| import java.util.Map; | |||
| import java.util.concurrent.CompletableFuture; | |||
| import java.util.concurrent.TimeUnit; | |||
| public class NSSystemLocation | |||
| { | |||
| private static final String ID_TAG = NSSystemLocation.class.getName(); | |||
| public static final String FUSED_PROVIDER = "fused"; | |||
| private final Context m_context; | |||
| private NSLocationInfo m_lastLocation = null; | |||
| private volatile CancellationSignal m_cancellationSignal; | |||
| private volatile CompletableFuture<Location> m_future; | |||
| public NSSystemLocation(Context context) | |||
| { | |||
| m_context = context; | |||
| } | |||
| public NSLocationInfo GetLocation(String provider) | |||
| { | |||
| NSLocationInfo loc; | |||
| try | |||
| { | |||
| loc = GetHighLocation(provider); | |||
| if(null == loc) | |||
| loc = GetLastLocation(provider); | |||
| if(null != loc) | |||
| m_lastLocation = loc; | |||
| else | |||
| ShowToast("定位失败, 请先开启位置服务", Toast.LENGTH_LONG); | |||
| return loc; | |||
| } | |||
| catch(Throwable e) | |||
| { | |||
| e.printStackTrace(); | |||
| throw new RuntimeException(e); | |||
| } | |||
| } | |||
| @SuppressLint("MissingPermission") | |||
| private NSLocationInfo GetLastLocation(final String type) | |||
| { | |||
| NSLocationInfo loc = null; | |||
| try | |||
| { | |||
| LocationManager lm = (LocationManager) m_context.getSystemService(Context.LOCATION_SERVICE); | |||
| List<String> allProviders = lm.getAllProviders(); | |||
| Log.i(ID_TAG, "使用最近所有定位提供器: " + allProviders); | |||
| Map<String, NSLocationInfo> map = new HashMap<>(); | |||
| for(String provider : allProviders) | |||
| { | |||
| Location lastKnownLocation = lm.getLastKnownLocation(provider); | |||
| if(null == lastKnownLocation) | |||
| continue; | |||
| NSLocationInfo l = new NSLocationInfo(provider, lastKnownLocation.getLongitude(), lastKnownLocation.getLatitude()); | |||
| map.put(provider, l); | |||
| } | |||
| Log.i(ID_TAG, "最近定位提供器结果: " + map); | |||
| if(null != type && !type.isEmpty() && map.containsKey(type)) | |||
| loc = map.get(type); | |||
| else if(map.containsKey(LocationManager.GPS_PROVIDER)) | |||
| loc = map.get(LocationManager.GPS_PROVIDER); | |||
| else if(map.containsKey(LocationManager.NETWORK_PROVIDER)) | |||
| loc = map.get(LocationManager.NETWORK_PROVIDER); | |||
| else if(map.containsKey(FUSED_PROVIDER)) | |||
| loc = map.get(FUSED_PROVIDER); | |||
| else if(map.containsKey(LocationManager.PASSIVE_PROVIDER)) | |||
| loc = map.get(LocationManager.PASSIVE_PROVIDER); | |||
| if(null == loc) | |||
| Log.i(ID_TAG, "所有提供器无最近定位"); | |||
| else | |||
| Log.i(ID_TAG, "使用提供器最近定位: " + loc.provider); | |||
| } | |||
| catch(Throwable e) | |||
| { | |||
| e.printStackTrace(); | |||
| } | |||
| return loc; | |||
| } | |||
| @SuppressLint("MissingPermission") | |||
| private NSLocationInfo GetHighLocation(String provider) | |||
| public NSLocationInfo Read(String provider, int timeout) | |||
| { | |||
| if(Build.VERSION.SDK_INT < Build.VERSION_CODES.R) | |||
| { | |||
| Log.w(ID_TAG, "获取当前定位请求Android 11+"); | |||
| return null; | |||
| } | |||
| @@ -153,9 +90,16 @@ public class NSSystemLocation | |||
| return null; | |||
| } | |||
| CompletableFuture<Location> future = new CompletableFuture<>(); | |||
| lm.getCurrentLocation(provider, null, m_context.getMainExecutor(), future::complete); | |||
| Location location = future.get(); | |||
| m_cancellationSignal = new CancellationSignal(); | |||
| m_future = new CompletableFuture<>(); | |||
| lm.getCurrentLocation(provider, m_cancellationSignal, m_context.getMainExecutor(), m_future::complete); | |||
| Location location; | |||
| if(timeout > 0) | |||
| location = m_future.get(timeout, TimeUnit.MILLISECONDS); | |||
| else | |||
| location = m_future.get(); | |||
| m_cancellationSignal = null; | |||
| m_future = null; | |||
| if(null == location) | |||
| { | |||
| Log.i(ID_TAG, "无法使用高精度提供器定位"); | |||
| @@ -163,11 +107,17 @@ public class NSSystemLocation | |||
| } | |||
| loc = new NSLocationInfo(provider, location.getLongitude(), location.getLatitude()); | |||
| Log.i(ID_TAG, "使用高精度提供器获取定位: " + loc); | |||
| m_lastLocation = loc; | |||
| } | |||
| catch(Throwable e) | |||
| { | |||
| e.printStackTrace(); | |||
| } | |||
| finally | |||
| { | |||
| m_cancellationSignal = null; | |||
| m_future = null; | |||
| } | |||
| return loc; | |||
| } | |||
| @@ -182,6 +132,28 @@ public class NSSystemLocation | |||
| })); | |||
| } | |||
| public void TerminateRead() | |||
| { | |||
| CancellationSignal signal = m_cancellationSignal; | |||
| if(null != signal && !signal.isCanceled()) | |||
| { | |||
| Log.i(ID_TAG, "系统中止定位: 1. 取消定位"); | |||
| signal.cancel(); | |||
| } | |||
| CompletableFuture<Location> future = m_future; | |||
| if(null != future && !future.isCancelled()) | |||
| { | |||
| Log.i(ID_TAG, "系统中止定位: 2. 取消等待结果"); | |||
| future.cancel(true); | |||
| } | |||
| } | |||
| public void Shutdown() | |||
| { | |||
| Log.i(ID_TAG, "销毁系统定位"); | |||
| TerminateRead(); | |||
| } | |||
| public NSLocationInfo GetLastLocation() | |||
| { | |||
| return m_lastLocation; | |||
| @@ -37,16 +37,39 @@ public final class NSConstants | |||
| return BuildConfig.APP_VENDOR; | |||
| } | |||
| /** | |||
| * App图标 | |||
| */ | |||
| public static String AppIcon() | |||
| { | |||
| return BuildConfig.APP_ICON; | |||
| } | |||
| /** | |||
| * App更新下载地址 | |||
| * 可以在 /gradle.properties 里配置 appUpdateUrl=链接地址, 不要携带双引号 | |||
| * 也可以在命令行添加 -PappUpdateUrl="链接地址", 双引号可携带也可不携带 | |||
| */ | |||
| public static String AppUpdateUrl() | |||
| { | |||
| return BuildConfig.APP_UPDATE_URL; | |||
| } | |||
| public static boolean IsHttps() | |||
| { | |||
| return AppHomeUrl().startsWith("https://"); | |||
| } | |||
| public static boolean IsDebug() | |||
| { | |||
| return BuildConfig.DEBUG; | |||
| } | |||
| public static boolean IsAMapLocationEnabled() | |||
| { | |||
| return true; | |||
| } | |||
| // 偏好默认值 | |||
| public static final String DEFAULT_LOCATION_MODE = NSEnums.LocationMode.REALTIME; | |||
| public static final int DEFAULT_LOCATION_GAODE_INTERVAL = 1000; | |||
| @@ -60,5 +83,10 @@ public final class NSConstants | |||
| public static final boolean DEFAULT_LOGCAT_CONSOLE_OUTPUT = false; | |||
| public static final int DEFAULT_LOCATION_TIMEOUT = 0; | |||
| public static final String FUSED_PROVIDER = "fused"; | |||
| public static final String APK_UPDATE_DOWNLOAD_DIR = "update"; | |||
| public static final String APK_UPDATE_DOWNLOAD_FILE = "latest.apk"; | |||
| private NSConstants() {} | |||
| } | |||
| @@ -19,10 +19,11 @@ public final class NSPreference | |||
| public static final String LAST_ACCESS_URL = "last_access_url"; | |||
| public static final String COOKIES = "cookies"; | |||
| public static final String FONT_SCALE = "font_scale"; | |||
| public static final String LOGCAT_CONSOLE_OUTPUT = "logcat_console_output"; | |||
| public static final String LOGCAT_CONSOLE_OUTPUT = "logcat_console_output"; | |||
| public static final String LOCATION_TIMEOUT = "location_timeout"; | |||
| public static final String RESET_SETTINGS = "RESET_SETTINGS"; | |||
| public static final String VERSION = "VERSION"; | |||
| public static final String UPDATE_DOWNLOAD = "UPDATE_DOWNLOAD"; | |||
| private final Context context; | |||
| @@ -2,8 +2,14 @@ package com.nsgk.ruralWeb.ui; | |||
| import android.content.Context; | |||
| import android.content.DialogInterface; | |||
| import android.content.Intent; | |||
| import android.content.SharedPreferences; | |||
| import android.net.Uri; | |||
| import android.os.Bundle; | |||
| import android.os.Handler; | |||
| import android.os.HandlerThread; | |||
| import android.os.Looper; | |||
| import android.provider.Settings; | |||
| import android.text.Html; | |||
| import android.util.Log; | |||
| import android.widget.Toast; | |||
| @@ -23,10 +29,13 @@ import com.nsgk.ruralWeb.utils.NSContextUtils; | |||
| import com.nsgk.ruralWeb.utils.NSMisc; | |||
| import com.nsgk.ruralWeb.utils.NSStr; | |||
| import java.io.File; | |||
| import java.util.Objects; | |||
| import cn.hutool.core.io.FileUtil; | |||
| import cn.hutool.core.util.NumberUtil; | |||
| import cn.hutool.core.util.StrUtil; | |||
| import cn.hutool.http.HttpUtil; | |||
| public class SettingsFragment extends PreferenceFragmentCompat implements Preference.OnPreferenceChangeListener, Preference.OnPreferenceClickListener | |||
| { | |||
| @@ -88,8 +97,17 @@ public class SettingsFragment extends PreferenceFragmentCompat implements Prefer | |||
| preference = findPreference(NSPreference.VERSION); | |||
| preference.setOnPreferenceClickListener(this); | |||
| preference = findPreference("location_gaode"); | |||
| preference.setVisible(false); | |||
| preference = findPreference(NSPreference.UPDATE_DOWNLOAD); | |||
| preference.setOnPreferenceClickListener(this); | |||
| if(StrUtil.isBlank(NSConstants.AppUpdateUrl())) | |||
| preference.setVisible(false); | |||
| // 总是隐藏高德的设置 | |||
| if(!NSConstants.IsAMapLocationEnabled() || !NSContextUtils.BuildIsDebug(context)) | |||
| { | |||
| preference = findPreference("location_gaode"); | |||
| preference.setVisible(false); | |||
| } | |||
| } | |||
| private boolean CheckValueIsNumber(Object newValue, Integer min) | |||
| @@ -164,7 +182,7 @@ public class SettingsFragment extends PreferenceFragmentCompat implements Prefer | |||
| break; | |||
| } | |||
| case NSPreference.LOCATION_MODE: | |||
| if(Objects.equals(NSEnums.LocationMode.GAODE, newValue)) | |||
| if(!NSConstants.IsAMapLocationEnabled() && Objects.equals(NSEnums.LocationMode.GAODE, newValue)) | |||
| { | |||
| Log.w(ID_TAG, "高德定位被禁用"); | |||
| return false; | |||
| @@ -202,10 +220,63 @@ public class SettingsFragment extends PreferenceFragmentCompat implements Prefer | |||
| case NSPreference.VERSION: | |||
| OpenAbout(); | |||
| break; | |||
| case NSPreference.UPDATE_DOWNLOAD: | |||
| DownloadUpdate(); | |||
| //DownloadUpdateExternally(); | |||
| break; | |||
| } | |||
| return false; | |||
| } | |||
| private void DownloadUpdateExternally() | |||
| { | |||
| String updateUrl = NSConstants.AppUpdateUrl(); | |||
| if(StrUtil.isBlank(updateUrl)) | |||
| return; | |||
| Context context = getContext(); | |||
| Toast.makeText(context, "正在打开下载地址......", Toast.LENGTH_SHORT).show(); | |||
| NSContextUtils.OpenUrlExternally(context, updateUrl); | |||
| } | |||
| private void DownloadUpdate() | |||
| { | |||
| String updateUrl = NSConstants.AppUpdateUrl(); | |||
| if(StrUtil.isBlank(updateUrl)) | |||
| return; | |||
| Handler handler = new Handler(Looper.myLooper()); | |||
| Context context = getContext(); | |||
| Log.d(ID_TAG, "下载apk地址: " + updateUrl); | |||
| Toast.makeText(context, "开始下载......", Toast.LENGTH_SHORT).show(); | |||
| // 创建文件夹并删除缓存文件 | |||
| File externalFilesDir = context.getExternalFilesDir(null); | |||
| String dir = externalFilesDir.getAbsolutePath() + File.separator + NSConstants.APK_UPDATE_DOWNLOAD_DIR; | |||
| FileUtil.mkdir(dir); | |||
| String file = dir + File.separator + NSConstants.APK_UPDATE_DOWNLOAD_FILE; | |||
| Log.d(ID_TAG, "删除文件: " + file); | |||
| FileUtil.del(file); | |||
| new Thread(() -> { | |||
| long bytes = HttpUtil.downloadFile(updateUrl, file); | |||
| Log.d(ID_TAG, "下载apk文件: " + FileUtil.readableFileSize(bytes)); | |||
| if(bytes > 0) | |||
| { | |||
| handler.post(() -> { | |||
| Toast.makeText(context, "准备安装......", Toast.LENGTH_SHORT).show(); | |||
| NSContextUtils.InstallApk(context, file); | |||
| }); | |||
| } | |||
| else | |||
| { | |||
| handler.post(() -> { | |||
| Toast.makeText(context, "下载更新失败", Toast.LENGTH_LONG).show(); | |||
| }); | |||
| } | |||
| }).start(); | |||
| } | |||
| private void ResetSettings() | |||
| { | |||
| NSPreference preference = new NSPreference(getContext()); | |||
| @@ -8,7 +8,15 @@ import android.content.pm.PackageInfo; | |||
| import android.content.pm.PackageManager; | |||
| import android.content.res.Resources; | |||
| import android.net.Uri; | |||
| import android.os.Build; | |||
| import android.provider.Settings; | |||
| import android.widget.Toast; | |||
| import androidx.core.content.FileProvider; | |||
| import com.nsgk.ruralWeb.sys.NSConstants; | |||
| import java.io.File; | |||
| public final class NSContextUtils | |||
| { | |||
| @@ -42,20 +50,6 @@ public final class NSContextUtils | |||
| return version; | |||
| } | |||
| public static boolean BuildIsDebug(Context context) | |||
| { | |||
| try | |||
| { | |||
| ApplicationInfo info = context.getApplicationInfo(); | |||
| return (info.flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0; | |||
| } | |||
| catch (Exception e) | |||
| { | |||
| e.printStackTrace(); | |||
| return false; // default is release | |||
| } | |||
| } | |||
| public static String GetListName(Context context, String value, int keyResource, int nameResource, String... def) | |||
| { | |||
| Resources resources = context.getResources(); | |||
| @@ -75,5 +69,54 @@ public final class NSContextUtils | |||
| return context.getResources().getString(id, args); | |||
| } | |||
| public static void OpenUrlExternally(Context context, String url) | |||
| { | |||
| Uri uri = Uri.parse(url); | |||
| Intent intent = new Intent(Intent.ACTION_VIEW, uri); | |||
| intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY | Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET); | |||
| context.startActivity(intent); | |||
| } | |||
| public static void InstallApk(Context context, String apkFile) | |||
| { | |||
| File file = new File(apkFile); | |||
| if(!file.isFile()) | |||
| return; | |||
| String packageName = context.getPackageName(); | |||
| Uri apkUri = FileProvider.getUriForFile(context, packageName + ".fileprovider", file); | |||
| Intent intent = new Intent(Intent.ACTION_VIEW); | |||
| intent.setDataAndType(apkUri, "application/vnd.android.package-archive"); | |||
| intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); | |||
| intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | |||
| if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { | |||
| if (!context.getPackageManager().canRequestPackageInstalls()) { | |||
| Toast.makeText(context, "Android 8以上安装apk请求允许未知来源App", Toast.LENGTH_LONG).show(); | |||
| Intent settingsIntent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, Uri.parse("package:" + packageName)); | |||
| context.startActivity(settingsIntent); | |||
| } else { | |||
| context.startActivity(intent); | |||
| } | |||
| } else { | |||
| context.startActivity(intent); | |||
| } | |||
| } | |||
| public static boolean BuildIsDebug(Context context) | |||
| { | |||
| try | |||
| { | |||
| ApplicationInfo info = context.getApplicationInfo(); | |||
| System.err.println(((info.flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0) + " -> " + NSConstants.IsDebug()); | |||
| return (info.flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0 || NSConstants.IsDebug(); | |||
| } | |||
| catch (Exception e) | |||
| { | |||
| e.printStackTrace(); | |||
| return NSConstants.IsDebug(); // default is release | |||
| } | |||
| } | |||
| private NSContextUtils() {} | |||
| } | |||
| @@ -20,6 +20,7 @@ import androidx.core.app.ActivityCompat; | |||
| import com.nsgk.ruralWeb.FullscreenActivity; | |||
| import com.nsgk.ruralWeb.R; | |||
| import com.nsgk.ruralWeb.enums.NSEnums; | |||
| import com.nsgk.ruralWeb.location.NSLastKnownLocation; | |||
| import com.nsgk.ruralWeb.location.NSLocationInfo; | |||
| import com.nsgk.ruralWeb.location.NSAMapLocation; | |||
| import com.nsgk.ruralWeb.location.NSRealtimeLocation; | |||
| @@ -31,41 +32,51 @@ import com.nsgk.ruralWeb.utils.NSContextUtils; | |||
| import java.util.ArrayList; | |||
| import java.util.Arrays; | |||
| import java.util.List; | |||
| import java.util.concurrent.atomic.AtomicInteger; | |||
| public class NSEnvWindowObject | |||
| { | |||
| private static final String ID_TAG = NSEnvWindowObject.class.getName(); | |||
| private static final String FUSED_PROVIDER = "fused"; | |||
| private final Handler m_handler; | |||
| private final Context m_context; | |||
| private final WebView m_webView; | |||
| private final NSSystemLocation location; | |||
| private final NSSystemLocation m_systemLocation; | |||
| private final NSLastKnownLocation m_lastKnownLocation; | |||
| private final NSAMapLocation m_amapLocation; | |||
| private final NSRealtimeLocation m_realtimeLocation; | |||
| private final NSAMapLocation amapLocation; | |||
| private final NSRealtimeLocation realtimeLocation; | |||
| private final NSPreference preference; | |||
| private final NSPreference m_preference; | |||
| // 最近一次获取到的定位坐标, 如果获取定位失败则返回此值. 可能不需要, 因为PASSIVE_PROVIDER就为最近的定位 | |||
| private String lastLocation = null; | |||
| private String m_lastLocation = null; | |||
| private static final int LOCATION_STATE_READY = 0; // 准备 | |||
| private static final int LOCATION_STATE_PROCESSING = 1; // 进行中 | |||
| private static final int LOCATION_STATE_FINISH = 2; // 完成 | |||
| private static final int LOCATION_STATE_TERMINATED = 3; // 中止 | |||
| private AtomicInteger m_locationState = new AtomicInteger(LOCATION_STATE_READY); | |||
| public NSEnvWindowObject(Context context, Handler handler, WebView webView) | |||
| { | |||
| m_context = context; | |||
| m_handler = handler; | |||
| m_webView = webView; | |||
| location = new NSSystemLocation(m_context); | |||
| amapLocation = new NSAMapLocation(m_context); | |||
| realtimeLocation = new NSRealtimeLocation(m_context); | |||
| preference = new NSPreference(m_context); | |||
| m_systemLocation = new NSSystemLocation(m_context); | |||
| m_lastKnownLocation = new NSLastKnownLocation(m_context); | |||
| m_amapLocation = new NSAMapLocation(m_context); | |||
| m_realtimeLocation = new NSRealtimeLocation(m_context); | |||
| m_preference = new NSPreference(m_context); | |||
| } | |||
| public void OnDestroy() | |||
| { | |||
| if(amapLocation.IsInitialized()) | |||
| amapLocation.Shutdown(); | |||
| if(realtimeLocation.IsInitialized()) | |||
| realtimeLocation.Shutdown(); | |||
| if(m_amapLocation.IsInitialized()) | |||
| m_amapLocation.Shutdown(); | |||
| if(m_realtimeLocation.IsInitialized()) | |||
| m_realtimeLocation.Shutdown(); | |||
| m_systemLocation.Shutdown(); | |||
| } | |||
| @JavascriptInterface | |||
| @@ -76,12 +87,8 @@ public class NSEnvWindowObject | |||
| private void ShowToast(final String message, int duration) | |||
| { | |||
| RunOnUIThread(new Runnable() { | |||
| @Override | |||
| public void run() | |||
| { | |||
| Toast.makeText(m_context, message, duration).show(); | |||
| } | |||
| RunOnUIThread(() -> { | |||
| Toast.makeText(m_context, message, duration).show(); | |||
| }); | |||
| } | |||
| @@ -98,37 +105,44 @@ public class NSEnvWindowObject | |||
| if(!CheckLocationPermission()) | |||
| { | |||
| return lastLocation; | |||
| return m_lastLocation; | |||
| } | |||
| SetLocationState(LOCATION_STATE_READY); | |||
| ShowIndicator(); | |||
| NSLocationInfo loc; | |||
| String location = CurrentLocationMode(); | |||
| String provider = preference.GetString(NSPreference.LOCATION_PROVIDER, NSConstants.DEFAULT_LOCATION_PROVIDER); | |||
| String provider = m_preference.GetString(NSPreference.LOCATION_PROVIDER, NSConstants.DEFAULT_LOCATION_PROVIDER); | |||
| Log.d(ID_TAG, "定位提供器: " + provider); | |||
| SetLocationState(LOCATION_STATE_PROCESSING); | |||
| switch(location) | |||
| { | |||
| case NSEnums.LocationMode.GAODE: | |||
| loc = GetAMapLocation(); | |||
| if(null == loc) | |||
| loc = GetBuiltInLocation(provider); | |||
| break; | |||
| case NSEnums.LocationMode.REALTIME: | |||
| loc = GetRealtimeLocation(); | |||
| if(null == loc) | |||
| loc = GetBuiltInLocation(provider); | |||
| break; | |||
| case NSEnums.LocationMode.SYSTEM: | |||
| default: | |||
| loc = GetBuiltInLocation(provider); | |||
| loc = GetCurrentLocation(provider); | |||
| break; | |||
| } | |||
| SetLocationState(LOCATION_STATE_FINISH, LOCATION_STATE_PROCESSING); | |||
| if(null == loc) | |||
| { | |||
| if(IsLocationState(LOCATION_STATE_FINISH)) | |||
| ShowToast("定位失败, 请先开启位置服务", Toast.LENGTH_LONG); | |||
| loc = GetLastKnownLocation(provider); | |||
| } | |||
| HideIndicator(); | |||
| return SetLastLocation(loc); | |||
| } | |||
| public String CurrentLocationMode() | |||
| { | |||
| return preference.GetString(NSPreference.LOCATION_MODE, NSConstants.DEFAULT_LOCATION_MODE); | |||
| return m_preference.GetString(NSPreference.LOCATION_MODE, NSConstants.DEFAULT_LOCATION_MODE); | |||
| } | |||
| @JavascriptInterface | |||
| @@ -140,7 +154,7 @@ public class NSEnvWindowObject | |||
| @JavascriptInterface | |||
| public int GetLocationTimeout() | |||
| { | |||
| return preference.GetIntFromString(NSPreference.LOCATION_TIMEOUT, NSConstants.DEFAULT_LOCATION_TIMEOUT); | |||
| return m_preference.GetIntFromString(NSPreference.LOCATION_TIMEOUT, NSConstants.DEFAULT_LOCATION_TIMEOUT); | |||
| } | |||
| @JavascriptInterface | |||
| @@ -182,7 +196,7 @@ public class NSEnvWindowObject | |||
| } | |||
| } | |||
| preference.SetString(NSPreference.LOCATION_MODE, mode); | |||
| m_preference.SetString(NSPreference.LOCATION_MODE, mode); | |||
| res[0] = mode; | |||
| String name = locationModeList.get(selected[0]); | |||
| Toast.makeText(m_context, "使用" + name + "定位", Toast.LENGTH_SHORT).show(); | |||
| @@ -192,7 +206,7 @@ public class NSEnvWindowObject | |||
| }); | |||
| builder.setNeutralButton("默认", (DialogInterface dialog, int which) -> { | |||
| res[0] = NSConstants.DEFAULT_LOCATION_MODE; | |||
| preference.SetString(NSPreference.LOCATION_MODE, res[0]); | |||
| m_preference.SetString(NSPreference.LOCATION_MODE, res[0]); | |||
| int choose = locationModeValueList.indexOf(res[0]); | |||
| String name = locationModeList.get(choose); | |||
| Toast.makeText(m_context, "使用" + name + "定位", Toast.LENGTH_SHORT).show(); | |||
| @@ -225,20 +239,21 @@ public class NSEnvWindowObject | |||
| private String SetLastLocation(NSLocationInfo loc) | |||
| { | |||
| if(null != loc) | |||
| lastLocation = ReturnLocation(loc); | |||
| return lastLocation; | |||
| m_lastLocation = ReturnLocation(loc); | |||
| return m_lastLocation; | |||
| } | |||
| private NSLocationInfo GetBuiltInLocation(String type) | |||
| private NSLocationInfo GetCurrentLocation(String type) | |||
| { | |||
| Log.i(ID_TAG, "使用系统定位"); | |||
| Log.i(ID_TAG, "使用系统当前定位"); | |||
| NSLocationInfo loc = null; | |||
| try | |||
| { | |||
| int timeout = m_preference.GetIntFromString(NSPreference.LOCATION_TIMEOUT, NSConstants.DEFAULT_LOCATION_TIMEOUT); | |||
| long start = System.currentTimeMillis(); | |||
| loc = location.GetLocation(type); | |||
| loc = m_systemLocation.Read(type, timeout); | |||
| long end = System.currentTimeMillis(); | |||
| Log.i(ID_TAG, "系统定位结果: " + loc + ", 耗时=" + (end - start) + "毫秒"); | |||
| Log.i(ID_TAG, "系统当前定位结果: " + loc + ", 耗时=" + (end - start) + "毫秒"); | |||
| } | |||
| catch(Throwable e) | |||
| { | |||
| @@ -247,6 +262,24 @@ public class NSEnvWindowObject | |||
| return loc; | |||
| } | |||
| private NSLocationInfo GetLastKnownLocation(String type) | |||
| { | |||
| Log.i(ID_TAG, "使用系统最近定位"); | |||
| NSLocationInfo loc; | |||
| try | |||
| { | |||
| long start = System.currentTimeMillis(); | |||
| loc = m_lastKnownLocation.Read(type); | |||
| long end = System.currentTimeMillis(); | |||
| Log.i(ID_TAG, "系统定位最近结果: " + loc + ", 耗时=" + (end - start) + "毫秒"); | |||
| } | |||
| catch(Throwable e) | |||
| { | |||
| e.printStackTrace(); | |||
| } | |||
| return m_lastKnownLocation.GetLastLocation(); | |||
| } | |||
| private NSLocationInfo GetRealtimeLocation() | |||
| { | |||
| Log.i(ID_TAG, "使用系统实时定位"); | |||
| @@ -254,10 +287,10 @@ public class NSEnvWindowObject | |||
| try | |||
| { | |||
| NSPreference preference = new NSPreference(m_context); | |||
| if(!realtimeLocation.IsInitialized()) | |||
| if(!m_realtimeLocation.IsInitialized()) | |||
| { | |||
| realtimeLocation.SetOption(NSRealtimeLocation.OPTION_MIN_TIME, preference.GetIntFromString(NSPreference.LOCATION_REALTIME_INTERVAL, NSConstants.DEFAULT_LOCATION_REALTIME_INTERVAL)); | |||
| realtimeLocation.SetOption(NSRealtimeLocation.OPTION_MIN_DISTANCE, preference.GetIntFromString(NSPreference.LOCATION_REALTIME_DISTANCE, NSConstants.DEFAULT_LOCATION_REALTIME_DISTANCE)); | |||
| m_realtimeLocation.SetOption(NSRealtimeLocation.OPTION_MIN_TIME, preference.GetIntFromString(NSPreference.LOCATION_REALTIME_INTERVAL, NSConstants.DEFAULT_LOCATION_REALTIME_INTERVAL)); | |||
| m_realtimeLocation.SetOption(NSRealtimeLocation.OPTION_MIN_DISTANCE, preference.GetIntFromString(NSPreference.LOCATION_REALTIME_DISTANCE, NSConstants.DEFAULT_LOCATION_REALTIME_DISTANCE)); | |||
| int flag = NSRealtimeLocation.DEFAULT_FLAG | NSRealtimeLocation.FLAG_BUILTIN_THREAD; | |||
| String provider = preference.GetString(NSPreference.LOCATION_PROVIDER, NSConstants.DEFAULT_LOCATION_PROVIDER); | |||
| if(LocationManager.GPS_PROVIDER.equals(provider)) | |||
| @@ -265,12 +298,12 @@ public class NSEnvWindowObject | |||
| flag &= ~NSRealtimeLocation.FLAG_PRIORITY_NETWORK; | |||
| flag |= NSRealtimeLocation.FLAG_PRIORITY_GPS; | |||
| } | |||
| realtimeLocation.Init(flag); | |||
| m_realtimeLocation.Init(flag); | |||
| } | |||
| long start = System.currentTimeMillis(); | |||
| int count = preference.GetIntFromString(NSPreference.LOCATION_REALTIME_READ_COUNT, NSConstants.DEFAULT_LOCATION_REALTIME_READ_COUNT); | |||
| int timeout = preference.GetIntFromString(NSPreference.LOCATION_TIMEOUT, NSConstants.DEFAULT_LOCATION_TIMEOUT); | |||
| loc = realtimeLocation.Read(count, timeout); | |||
| loc = m_realtimeLocation.Read(count, timeout); | |||
| long end = System.currentTimeMillis(); | |||
| Log.i(ID_TAG, "系统实时定位结果: " + loc + ", 耗时=" + (end - start) + "毫秒"); | |||
| } | |||
| @@ -288,15 +321,15 @@ public class NSEnvWindowObject | |||
| try | |||
| { | |||
| NSPreference preference = new NSPreference(m_context); | |||
| if(!amapLocation.IsInitialized()) | |||
| if(!m_amapLocation.IsInitialized()) | |||
| { | |||
| amapLocation.SetOption(NSAMapLocation.OPTION_INTERVAL, preference.GetIntFromString(NSPreference.LOCATION_GAODE_INTERVAL, NSConstants.DEFAULT_LOCATION_GAODE_INTERVAL)); | |||
| amapLocation.Init(NSAMapLocation.DEFAULT_FLAG | NSAMapLocation.FLAG_BUILTIN_THREAD); | |||
| m_amapLocation.SetOption(NSAMapLocation.OPTION_INTERVAL, preference.GetIntFromString(NSPreference.LOCATION_GAODE_INTERVAL, NSConstants.DEFAULT_LOCATION_GAODE_INTERVAL)); | |||
| m_amapLocation.Init(NSAMapLocation.DEFAULT_FLAG | NSAMapLocation.FLAG_BUILTIN_THREAD); | |||
| } | |||
| long start = System.currentTimeMillis(); | |||
| int count = preference.GetIntFromString(NSPreference.LOCATION_GAODE_READ_COUNT, NSConstants.DEFAULT_LOCATION_GAODE_READ_COUNT); | |||
| int timeout = preference.GetIntFromString(NSPreference.LOCATION_TIMEOUT, NSConstants.DEFAULT_LOCATION_TIMEOUT); | |||
| loc = amapLocation.Read(count, timeout); | |||
| loc = m_amapLocation.Read(count, timeout); | |||
| long end = System.currentTimeMillis(); | |||
| Log.i(ID_TAG, "高德定位结果: " + loc + ", 耗时=" + (end - start) + "毫秒"); | |||
| } | |||
| @@ -351,19 +384,15 @@ public class NSEnvWindowObject | |||
| sb.append(", "); | |||
| } | |||
| final String arg = sb.toString(); | |||
| RunOnUIThread(new Runnable() { | |||
| @Override | |||
| public void run() | |||
| { | |||
| String script = "javascript:typeof(" + func + ") == 'function' && " + func + "(" + arg + ");"; | |||
| Log.e(ID_TAG, String.format("调用js函数: 函数(%s), 参数(%s)", func, arg)); | |||
| m_webView.evaluateJavascript(script, new ValueCallback<String>() { | |||
| @Override | |||
| public void onReceiveValue(String value) { | |||
| Log.e(ID_TAG, String.format("js函数 %s 返回: %s", func, value)); | |||
| } | |||
| }); | |||
| } | |||
| RunOnUIThread(() -> { | |||
| String script = "javascript:typeof(" + func + ") == 'function' && " + func + "(" + arg + ");"; | |||
| Log.e(ID_TAG, String.format("调用js函数: 函数(%s), 参数(%s)", func, arg)); | |||
| m_webView.evaluateJavascript(script, new ValueCallback<String>() { | |||
| @Override | |||
| public void onReceiveValue(String value) { | |||
| Log.e(ID_TAG, String.format("js函数 %s 返回: %s", func, value)); | |||
| } | |||
| }); | |||
| }); | |||
| } | |||
| @@ -377,14 +406,63 @@ public class NSEnvWindowObject | |||
| sb.append(", "); | |||
| } | |||
| final String arg = sb.toString(); | |||
| RunOnUIThread(new Runnable() { | |||
| @Override | |||
| public void run() { | |||
| Log.e(ID_TAG, String.format("调用js函数: 函数(%s), 参数(%s)", func, arg)); | |||
| String script = "javascript:typeof(" + func + ") == 'function' && " + func + "(" + arg + ");"; | |||
| Log.e(ID_TAG, script); | |||
| m_webView.loadUrl(script, null); | |||
| } | |||
| RunOnUIThread(() -> { | |||
| Log.e(ID_TAG, String.format("调用js函数: 函数(%s), 参数(%s)", func, arg)); | |||
| String script = "javascript:typeof(" + func + ") == 'function' && " + func + "(" + arg + ");"; | |||
| Log.e(ID_TAG, script); | |||
| m_webView.loadUrl(script, null); | |||
| }); | |||
| } | |||
| protected void ShowIndicator() | |||
| { | |||
| RunOnUIThread(() -> { | |||
| ((FullscreenActivity)(m_context)).SetIndicatorVisible(true); | |||
| }); | |||
| } | |||
| protected void HideIndicator() | |||
| { | |||
| RunOnUIThread(() -> { | |||
| ((FullscreenActivity)(m_context)).SetIndicatorVisible(false); | |||
| }); | |||
| } | |||
| public void TerminateLocation() | |||
| { | |||
| if(!IsLocationState(LOCATION_STATE_PROCESSING)) | |||
| return; | |||
| SetLocationState(LOCATION_STATE_TERMINATED, LOCATION_STATE_PROCESSING); | |||
| String location = CurrentLocationMode(); | |||
| Log.i(ID_TAG, "中止定位: " + location); | |||
| switch(location) | |||
| { | |||
| case NSEnums.LocationMode.GAODE: | |||
| m_amapLocation.TerminateRead(); | |||
| break; | |||
| case NSEnums.LocationMode.REALTIME: | |||
| m_realtimeLocation.TerminateRead(); | |||
| break; | |||
| case NSEnums.LocationMode.SYSTEM: | |||
| m_systemLocation.TerminateRead(); | |||
| break; | |||
| } | |||
| HideIndicator(); | |||
| } | |||
| private void SetLocationState(int st) | |||
| { | |||
| m_locationState.set(st); | |||
| } | |||
| private void SetLocationState(int st, int cur) | |||
| { | |||
| m_locationState.compareAndSet(cur, st); | |||
| } | |||
| private boolean IsLocationState(int st) | |||
| { | |||
| return m_locationState.get() == st; | |||
| } | |||
| } | |||
| @@ -0,0 +1,12 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | |||
| <shape xmlns:android="http://schemas.android.com/apk/res/android" > | |||
| <solid android:color="@color/black_overlay" /> | |||
| <padding | |||
| android:bottom="10dp" | |||
| android:left="20dp" | |||
| android:right="20dp" | |||
| android:top="10dp" /> | |||
| <corners android:radius="10dp" /> | |||
| </shape> | |||
| @@ -0,0 +1,24 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | |||
| <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" | |||
| android:orientation="vertical" | |||
| android:layout_width="wrap_content" | |||
| android:layout_height="wrap_content" | |||
| android:background="@drawable/indicator_background"> | |||
| <ProgressBar | |||
| android:id="@+id/progress_indicator" | |||
| android:layout_width="wrap_content" | |||
| android:layout_height="wrap_content" | |||
| android:layout_gravity="center_horizontal" | |||
| android:indeterminate="true" | |||
| /> | |||
| <Button | |||
| android:id="@+id/terminate_button" | |||
| android:layout_width="wrap_content" | |||
| android:layout_height="wrap_content" | |||
| android:layout_gravity="center_horizontal" | |||
| android:layout_marginTop="2dp" | |||
| android:text="停止" /> | |||
| </LinearLayout> | |||
| @@ -6,14 +6,14 @@ | |||
| <item>系统(实时)</item> | |||
| <item>系统</item> | |||
| <item>H5</item> | |||
| <!--<item>高德</item>--> | |||
| <item>高德</item> | |||
| </string-array> | |||
| <string-array name="location_mode_values"> | |||
| <item>realtime</item> | |||
| <item>system</item> | |||
| <item>h5</item> | |||
| <!--<item>gaode</item>--> | |||
| <item>gaode</item> | |||
| </string-array> | |||
| <string-array name="location_provider_labels"> | |||
| @@ -131,6 +131,13 @@ | |||
| android:persistent="false"> | |||
| </Preference> | |||
| <Preference | |||
| android:key="UPDATE_DOWNLOAD" | |||
| android:summary="下载更新" | |||
| android:title="下载更新" | |||
| android:persistent="false"> | |||
| </Preference> | |||
| <Preference | |||
| android:key="VERSION" | |||
| android:title="版本" | |||
| @@ -32,4 +32,8 @@ appIconKey=yhzl | |||
| # [String] App copyright name, it will save to @string/app_copyright. default using @string/copyright | |||
| appCopyright= | |||
| # [String] App vendor name, it will save to @string/app_vendor. default using @string/company | |||
| appVendor= | |||
| appVendor= | |||
| # [String] App update download url, it will save to @string/app_update_url | |||
| appUpdateUrl=http://218.59.175.43:8090/nsgk/qixingguan.apk | |||
| # gaode amap key | |||
| amapKey=490bef43ef0182379aa1a8bacc45d054 | |||