From 9ead47bdf7e0c35b6e44de34af9df22ddff55a23 Mon Sep 17 00:00:00 2001 From: Jens Date: Sun, 3 Dec 2023 23:24:58 +0100 Subject: [PATCH 01/50] Permissions for startActivity() reduced --- .../java/com/jens/automation2/Actions.java | 7 +++- .../ActivityManageActionStartActivity.java | 34 +++++++++++-------- .../jens/automation2/ActivityPermissions.java | 13 ++++++- .../jens/automation2/XmlFileInterface.java | 14 +++++--- .../receivers/BatteryReceiver.java | 2 +- .../metadata/android/en-US/changelogs/138.txt | 1 + 6 files changed, 49 insertions(+), 22 deletions(-) create mode 100644 fastlane/metadata/android/en-US/changelogs/138.txt diff --git a/app/src/main/java/com/jens/automation2/Actions.java b/app/src/main/java/com/jens/automation2/Actions.java index bc830e2e..16037258 100644 --- a/app/src/main/java/com/jens/automation2/Actions.java +++ b/app/src/main/java/com/jens/automation2/Actions.java @@ -1043,7 +1043,12 @@ public class Actions { Miscellaneous.logEvent("i", "StartOtherActivity", "Starting other Activity...", 4); - String params[] = param.split(";"); + String params[]; + + if(param.contains(Action.actionParameter2Split)) + params = param.split(Action.actionParameter2Split); + else + params = param.split(";"); try { diff --git a/app/src/main/java/com/jens/automation2/ActivityManageActionStartActivity.java b/app/src/main/java/com/jens/automation2/ActivityManageActionStartActivity.java index 08072684..7c179ef8 100644 --- a/app/src/main/java/com/jens/automation2/ActivityManageActionStartActivity.java +++ b/app/src/main/java/com/jens/automation2/ActivityManageActionStartActivity.java @@ -58,10 +58,10 @@ public class ActivityManageActionStartActivity extends Activity RadioButton rbStartAppSelectByActivity, rbStartAppSelectByAction, rbStartAppByActivity, rbStartAppByBroadcast, rbStartAppByService, rbStartAppByForegroundService; final String urlShowExamples = "https://server47.de/automation/examples_startProgram.html"; - final static String startByActivityString = "0"; - final static String startByBroadcastString = "1"; - final static String startByServiceString = "2"; - final static String startByForegroundServiceString = "3"; + public final static String startByActivityString = "0"; + public final static String startByBroadcastString = "1"; + public final static String startByServiceString = "2"; + public final static String startByForegroundServiceString = "3"; final static int requestCodeForRequestQueryAllPackagesPermission = 4711; @@ -234,29 +234,29 @@ public class ActivityManageActionStartActivity extends Activity String parameter2 = ""; if (rbStartAppSelectByActivity.isChecked()) - parameter2 += etPackageName.getText().toString() + ";" + etActivityOrActionPath.getText().toString(); + parameter2 += etPackageName.getText().toString() + Action.actionParameter2Split + etActivityOrActionPath.getText().toString(); else { if (etPackageName.getText().toString() != null && etPackageName.getText().toString().length() > 0) - parameter2 += etPackageName.getText().toString() + ";" + etActivityOrActionPath.getText().toString(); + parameter2 += etPackageName.getText().toString() + Action.actionParameter2Split + etActivityOrActionPath.getText().toString(); else - parameter2 += Actions.dummyPackageString + ";" + etActivityOrActionPath.getText().toString(); + parameter2 += Actions.dummyPackageString + Action.actionParameter2Split + etActivityOrActionPath.getText().toString(); // if(etClassName.getText().toString().length() > 0) - parameter2 += ";" + etClassName.getText().toString(); + parameter2 += Action.actionParameter2Split + etClassName.getText().toString(); } if (rbStartAppByActivity.isChecked()) - parameter2 += ";" + startByActivityString; + parameter2 += Action.actionParameter2Split + startByActivityString; else if(rbStartAppByService.isChecked()) - parameter2 += ";" + startByServiceString; + parameter2 += Action.actionParameter2Split + startByServiceString; else if(rbStartAppByForegroundService.isChecked()) - parameter2 += ";" + startByForegroundServiceString; + parameter2 += Action.actionParameter2Split + startByForegroundServiceString; else - parameter2 += ";" + startByBroadcastString; + parameter2 += Action.actionParameter2Split + startByBroadcastString; for (String s : intentPairList) - parameter2 += ";" + s; + parameter2 += Action.actionParameter2Split + s; returnData.putExtra(ActivityManageRule.intentNameActionParameter2, parameter2); @@ -628,7 +628,13 @@ public class ActivityManageActionStartActivity extends Activity rbStartAppSelectByActivity.setChecked(!selectionByAction); rbStartAppSelectByAction.setChecked(selectionByAction); - String[] params = input.getStringExtra(ActivityManageRule.intentNameActionParameter2).split(";"); + String[] params; + String partsString = input.getStringExtra(ActivityManageRule.intentNameActionParameter2); + + if(partsString.contains(Action.actionParameter2Split)) + params = partsString.split(Action.actionParameter2Split); + else + params = partsString.split(";"); if(Miscellaneous.isNumeric(params[2])) // old configuration file { diff --git a/app/src/main/java/com/jens/automation2/ActivityPermissions.java b/app/src/main/java/com/jens/automation2/ActivityPermissions.java index 6bf0f004..f18c2238 100644 --- a/app/src/main/java/com/jens/automation2/ActivityPermissions.java +++ b/app/src/main/java/com/jens/automation2/ActivityPermissions.java @@ -650,7 +650,18 @@ public class ActivityPermissions extends Activity // ) // addToArrayListUnique("net.kollnig.missioncontrol.permission.ADMIN", requiredPermissions); if(Build.VERSION.SDK_INT >= 29) - addToArrayListUnique(Manifest.permission.SYSTEM_ALERT_WINDOW, requiredPermissions); + { + String parts[]; + if(action.getParameter2().contains(Action.actionParameter2Split)) + parts = action.getParameter2().split(Action.actionParameter2Split); + else + parts = action.getParameter2().split(";"); + + // Permission only required for starts of activity, not broadcasts or services + + if(parts[2].equals(ActivityManageActionStartActivity.startByActivityString)) + addToArrayListUnique(Manifest.permission.SYSTEM_ALERT_WINDOW, requiredPermissions); + } break; case triggerUrl: addToArrayListUnique(Manifest.permission.INTERNET, requiredPermissions); diff --git a/app/src/main/java/com/jens/automation2/XmlFileInterface.java b/app/src/main/java/com/jens/automation2/XmlFileInterface.java index ca192c93..aa3bddb5 100644 --- a/app/src/main/java/com/jens/automation2/XmlFileInterface.java +++ b/app/src/main/java/com/jens/automation2/XmlFileInterface.java @@ -1286,20 +1286,24 @@ public class XmlFileInterface else newTag = tag.replace("/", Action.intentPairSeparator); - String[] newTagPieces = newTag.split(";"); + String[] newTagPieces = new String[0]; + if(newTag.contains(Action.actionParameter2Split)) + newTagPieces = newTag.split(Action.actionParameter2Split); + else + newTag.split(";"); if(newTagPieces.length < 2 || (!newTagPieces[0].contains(Actions.dummyPackageString) && newTagPieces[1].contains(Action.intentPairSeparator))) { - newTag = Actions.dummyPackageString + ";" + newTag; - newTagPieces = newTag.split(";"); + newTag = Actions.dummyPackageString + Action.actionParameter2Split + newTag; + newTagPieces = newTag.split(Action.actionParameter2Split); } if(newTagPieces.length < 3) - newTag += ";" + ActivityManageActionStartActivity.startByActivityString; + newTag += Action.actionParameter2Split + ActivityManageActionStartActivity.startByActivityString; else if(newTagPieces.length >= 3) { if(newTagPieces[2].contains(Action.intentPairSeparator)) - newTag = newTagPieces[0] + ";" + newTagPieces[1] + ";" + ActivityManageActionStartActivity.startByActivityString + ";" + newTagPieces[2]; + newTag = newTagPieces[0] + Action.actionParameter2Split + newTagPieces[1] + Action.actionParameter2Split + ActivityManageActionStartActivity.startByActivityString + Action.actionParameter2Split + newTagPieces[2]; } newAction.setParameter2(newTag); diff --git a/app/src/main/java/com/jens/automation2/receivers/BatteryReceiver.java b/app/src/main/java/com/jens/automation2/receivers/BatteryReceiver.java index d9064e11..6baaeb58 100644 --- a/app/src/main/java/com/jens/automation2/receivers/BatteryReceiver.java +++ b/app/src/main/java/com/jens/automation2/receivers/BatteryReceiver.java @@ -134,7 +134,7 @@ public class BatteryReceiver extends BroadcastReceiver implements AutomationList { case BatteryManager.BATTERY_STATUS_CHARGING: case BatteryManager.BATTERY_STATUS_FULL: - Miscellaneous.logEvent("i", "BatteryReceiver", "Device has been fully charged.", 5); +// Miscellaneous.logEvent("i", "BatteryReceiver", "Device has been fully charged.", 5); this.actionCharging(context); break; case BatteryManager.BATTERY_STATUS_DISCHARGING: diff --git a/fastlane/metadata/android/en-US/changelogs/138.txt b/fastlane/metadata/android/en-US/changelogs/138.txt new file mode 100644 index 00000000..5bed4132 --- /dev/null +++ b/fastlane/metadata/android/en-US/changelogs/138.txt @@ -0,0 +1 @@ +* Fixed: Overlay permission for start other program action only required if startByActivity() is selected \ No newline at end of file From 3c8c0f14f2f2e60096976adf6362bf14423003ca Mon Sep 17 00:00:00 2001 From: Jens Date: Wed, 6 Dec 2023 23:44:40 +0100 Subject: [PATCH 02/50] Fixed bug in broadcast receiver trigger --- .../com/jens/automation2/receivers/BroadcastListener.java | 8 ++++++-- fastlane/metadata/android/en-US/changelogs/138.txt | 3 ++- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/app/src/main/java/com/jens/automation2/receivers/BroadcastListener.java b/app/src/main/java/com/jens/automation2/receivers/BroadcastListener.java index 204a3b5d..b994a3e7 100644 --- a/app/src/main/java/com/jens/automation2/receivers/BroadcastListener.java +++ b/app/src/main/java/com/jens/automation2/receivers/BroadcastListener.java @@ -52,9 +52,13 @@ public class BroadcastListener extends android.content.BroadcastReceiver impleme { broadcastsCollection.add(new EventOccurrence(Calendar.getInstance(), intent.getAction())); - for(String key : intent.getExtras().keySet()) + Miscellaneous.logEvent("i", "Broadcast received", "Broadcast " + intent.getAction() + " received.", 4); + if(intent.getExtras() != null && intent.getExtras().size() > 0) { - Miscellaneous.logEvent("i", "Broadcast extra", "Broadcast " + intent.getAction() + " has extra " + key + " and type " + intent.getExtras().get(key).getClass().getName(), 4); + for (String key : intent.getExtras().keySet()) + { + Miscellaneous.logEvent("i", "Broadcast extra", "Broadcast " + intent.getAction() + " has extra " + key + " and type " + intent.getExtras().get(key).getClass().getName(), 4); + } } ArrayList ruleCandidates = Rule.findRuleCandidates(Trigger.Trigger_Enum.broadcastReceived); diff --git a/fastlane/metadata/android/en-US/changelogs/138.txt b/fastlane/metadata/android/en-US/changelogs/138.txt index 5bed4132..ae7eecd7 100644 --- a/fastlane/metadata/android/en-US/changelogs/138.txt +++ b/fastlane/metadata/android/en-US/changelogs/138.txt @@ -1 +1,2 @@ -* Fixed: Overlay permission for start other program action only required if startByActivity() is selected \ No newline at end of file +* Fixed: Overlay permission for start other program action only required if startByActivity() is selected +* Fixed: Broadcast receiver trigger would not trigger anything, but crash \ No newline at end of file From 1b8dc5de5f9f6dd3a3dab0edca07c39bc688a671 Mon Sep 17 00:00:00 2001 From: Jens Date: Mon, 11 Dec 2023 22:50:07 +0100 Subject: [PATCH 03/50] Http method selectable --- .../java/com/jens/automation2/Action.java | 23 +++++--- .../ActivityManageActionTriggerUrl.java | 58 ++++++++++++++++--- .../com/jens/automation2/Miscellaneous.java | 38 ++++++++---- .../jens/automation2/XmlFileInterface.java | 2 +- .../activity_manage_action_trigger_url.xml | 28 +++++++++ app/src/main/res/values/strings.xml | 3 + 6 files changed, 124 insertions(+), 28 deletions(-) diff --git a/app/src/main/java/com/jens/automation2/Action.java b/app/src/main/java/com/jens/automation2/Action.java index 0983fa9a..b6cf8d85 100644 --- a/app/src/main/java/com/jens/automation2/Action.java +++ b/app/src/main/java/com/jens/automation2/Action.java @@ -648,6 +648,7 @@ public class Action { String username = null; String password = null; + String method = ActivityManageActionTriggerUrl.methodGet; String url; String[] components = getParameter2().split(";"); @@ -657,6 +658,9 @@ public class Action username = components[0]; password = components[1]; url = components[2]; + + if(components.length >= 4) + method = components[3]; } else // compatibility for very old versions which haven't upgraded, yet. url = components[0]; @@ -670,7 +674,7 @@ public class Action Miscellaneous.logEvent("i", "HTTP", "Attempting download of " + url, 4); //getResources().getString("attemptingDownloadOf"); if(this.getParameter1()) // use authentication - new DownloadTask().execute(url, username, password); + new DownloadTask().execute(url, username, password, methodnochindiesignaturaufnehmen); else new DownloadTask().execute(url, null, null); } @@ -692,16 +696,19 @@ public class Action String urlUsername = null; String urlPassword = null; + String method = ActivityManageActionTriggerUrl.methodGet; if(parameters.length >= 3) { - urlUsername=parameters[1]; - urlPassword=parameters[2]; + urlUsername = parameters[1]; + urlPassword = parameters[2]; + + if(parameters.length >= 4) + method = parameters[3]; } String response = "httpError"; - HttpGet post; - - if(Settings.httpAttempts < 1) + + if(Settings.httpAttempts < 1) Miscellaneous.logEvent("w", "HTTP Request", Miscellaneous.getAnyContext().getResources().getString(R.string.cantDownloadTooFewRequestsInSettings), 3); while(attempts <= Settings.httpAttempts && response.equals("httpError")) @@ -710,9 +717,9 @@ public class Action // Either thorough checking or no encryption if(!Settings.httpAcceptAllCertificates || !urlString.toLowerCase(Locale.getDefault()).contains("https")) - response = Miscellaneous.downloadURL(urlString, urlUsername, urlPassword); + response = Miscellaneous.downloadURL(urlString, urlUsername, urlPassword, method); else - response = Miscellaneous.downloadURLwithoutCertificateChecking(urlString, urlUsername, urlPassword); + response = Miscellaneous.downloadURLwithoutCertificateChecking(urlString, urlUsername, urlPassword, method); try { diff --git a/app/src/main/java/com/jens/automation2/ActivityManageActionTriggerUrl.java b/app/src/main/java/com/jens/automation2/ActivityManageActionTriggerUrl.java index 4f1dc646..3849eced 100644 --- a/app/src/main/java/com/jens/automation2/ActivityManageActionTriggerUrl.java +++ b/app/src/main/java/com/jens/automation2/ActivityManageActionTriggerUrl.java @@ -13,6 +13,7 @@ import android.widget.CompoundButton; import android.widget.CompoundButton.OnCheckedChangeListener; import android.widget.EditText; import android.widget.ListView; +import android.widget.RadioButton; import android.widget.TableLayout; import android.widget.Toast; @@ -26,11 +27,15 @@ public class ActivityManageActionTriggerUrl extends Activity EditText etTriggerUrl, etTriggerUrlUsername, etTriggerUrlPassword; ListView lvTriggerUrlPostParameters; CheckBox chkTriggerUrlUseAuthentication; + RadioButton rbTriggerUrlMethodGet, rbTriggerUrlMethodPost; TableLayout tlTriggerUrlAuthentication; ArrayAdapter> lvTriggerUrlPostParametersAdapter; // private String existingUrl = ""; + + public static final String methodGet = "GET"; + public static final String methodPost = "POST"; public static boolean edit = false; public static Action resultingAction = null; @@ -49,6 +54,9 @@ public class ActivityManageActionTriggerUrl extends Activity lvTriggerUrlPostParameters = (ListView)findViewById(R.id.lvTriggerUrlPostParameters); tlTriggerUrlAuthentication = (TableLayout)findViewById(R.id.tlTriggerUrlAuthentication); bSaveTriggerUrl = (Button)findViewById(R.id.bSaveSpeakText); + rbTriggerUrlMethodGet = (RadioButton) findViewById(R.id.rbTriggerUrlMethodGet); + rbTriggerUrlMethodPost = (RadioButton) findViewById(R.id.rbTriggerUrlMethodPost); + bSaveTriggerUrl.setOnClickListener(new OnClickListener() { @Override @@ -59,6 +67,8 @@ public class ActivityManageActionTriggerUrl extends Activity if(resultingAction == null) { resultingAction = new Action(); + } + resultingAction.setAction(Action_Enum.triggerUrl); resultingAction.setParameter1(chkTriggerUrlUseAuthentication.isChecked()); @@ -70,13 +80,18 @@ public class ActivityManageActionTriggerUrl extends Activity if(password == null) password = ""; - + + String method = methodGet; + if(rbTriggerUrlMethodPost.isChecked()) + method = methodPost; + ActivityManageActionTriggerUrl.resultingAction.setParameter2( username + ";" + password + ";" + - etTriggerUrl.getText().toString().trim() + etTriggerUrl.getText().toString().trim() + ";" + + method ); - } + backToRuleManager(); } else @@ -84,16 +99,25 @@ public class ActivityManageActionTriggerUrl extends Activity } }); - chkTriggerUrlUseAuthentication.setOnCheckedChangeListener(new OnCheckedChangeListener() - { + { @Override public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { if(isChecked) + { tlTriggerUrlAuthentication.setVisibility(View.VISIBLE); + rbTriggerUrlMethodGet.setChecked(false); + rbTriggerUrlMethodPost.setChecked(true); + rbTriggerUrlMethodGet.setEnabled(false); + rbTriggerUrlMethodPost.setEnabled(false); + } else + { tlTriggerUrlAuthentication.setVisibility(View.GONE); + rbTriggerUrlMethodGet.setEnabled(true); + rbTriggerUrlMethodPost.setEnabled(true); + } etTriggerUrlUsername.setEnabled(isChecked); etTriggerUrlPassword.setEnabled(isChecked); @@ -109,7 +133,6 @@ public class ActivityManageActionTriggerUrl extends Activity } }); updateListView(); - ActivityManageActionTriggerUrl.edit = getIntent().getBooleanExtra("edit", false); if(edit) @@ -123,6 +146,20 @@ public class ActivityManageActionTriggerUrl extends Activity chkTriggerUrlUseAuthentication.setChecked(ActivityManageActionTriggerUrl.resultingAction.getParameter1()); etTriggerUrlUsername.setText(components[0]); etTriggerUrlPassword.setText(components[1]); + + if(components.length >= 4) + { + switch(components[3]) + { + case methodPost: + rbTriggerUrlMethodPost.setChecked(true); + break; + case methodGet: + default: + rbTriggerUrlMethodGet.setChecked(true); + break; + } + } } else etTriggerUrl.setText(components[0]); @@ -141,13 +178,18 @@ public class ActivityManageActionTriggerUrl extends Activity if(password == null) password = ""; - + + String method = methodGet; + if(rbTriggerUrlMethodPost.isChecked()) + method = methodPost; + ActivityManageActionTriggerUrl.resultingAction.setParameter1(chkTriggerUrlUseAuthentication.isChecked()); ActivityManageActionTriggerUrl.resultingAction.setParameter2( username + ";" + password + ";" + - etTriggerUrl.getText().toString() + etTriggerUrl.getText().toString() + ";" + + method ); } diff --git a/app/src/main/java/com/jens/automation2/Miscellaneous.java b/app/src/main/java/com/jens/automation2/Miscellaneous.java index 4275f26c..97bf03a3 100644 --- a/app/src/main/java/com/jens/automation2/Miscellaneous.java +++ b/app/src/main/java/com/jens/automation2/Miscellaneous.java @@ -44,7 +44,10 @@ import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.HttpVersion; import org.apache.http.client.HttpClient; +import org.apache.http.client.methods.HttpEntityEnclosingRequestBase; +import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPost; +import org.apache.http.client.methods.HttpPut; import org.apache.http.conn.ssl.SSLSocketFactory; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.params.BasicHttpParams; @@ -120,7 +123,7 @@ public class Miscellaneous extends Service public static final String lineSeparator = System.getProperty("line.separator"); - public static String downloadURL(String url, String username, String password) + public static String downloadURL(String url, String username, String password, String method) { HttpClient httpclient = new DefaultHttpClient(); StringBuilder responseBody = new StringBuilder(); @@ -148,6 +151,8 @@ public class Miscellaneous extends Service connection.setDoOutput(true); connection.setRequestProperty ("Authorization", "Basic " + encodedCredentials); } + else if(method.equals(ActivityManageActionTriggerUrl.methodPost)) + connection.setRequestMethod("POST"); InputStream content = (InputStream)connection.getInputStream(); BufferedReader in = new BufferedReader (new InputStreamReader (content)); @@ -174,10 +179,8 @@ public class Miscellaneous extends Service } } - public static String downloadURLwithoutCertificateChecking(String url, String username, String password) + public static String downloadURLwithoutCertificateChecking(String url, String username, String password, String method) { -// HttpClient httpclient = new DefaultHttpClient(); -// StringBuilder responseBody = new StringBuilder(); boolean errorFound = false; try @@ -187,20 +190,33 @@ public class Miscellaneous extends Service HttpProtocolParams.setVersion(params, HttpVersion.HTTP_1_1); HttpClient httpclient = new DefaultHttpClient(params); httpclient = Actions.getInsecureSslClient(httpclient); - - HttpPost httppost = new HttpPost(url); + + HttpEntityEnclosingRequestBase httpRequest; + switch (method) + { + case ActivityManageActionTriggerUrl.methodPost: + httpRequest = new HttpPost(url); + break; + case ActivityManageActionTriggerUrl.methodGet: + default: + httpRequest = new HttpEntityEnclosingRequestBase() + { + @Override + public String getMethod() + {somethingMissing + return null; + } + }; + } // Add http simple authentication if specified if(username != null && password != null) { String encodedCredentials = Base64.encodeToString(new String(username + ":" + password).getBytes(), Base64.DEFAULT); -// List nameValuePairs = new ArrayList(1); - httppost.addHeader("Authorization", "Basic " + encodedCredentials); -// nameValuePairs.add(new BasicNameValuePair("Authorization", "Basic " + encodedCredentials)); -// httppost.setEntity(new UrlEncodedFormEntity(nameValuePairs, "UTF-8")); + httpRequest.addHeader("Authorization", "Basic " + encodedCredentials); } - HttpResponse response = httpclient.execute(httppost); + HttpResponse response = httpclient.execute(httpRequest); HttpEntity entity = response.getEntity(); if (entity != null) { diff --git a/app/src/main/java/com/jens/automation2/XmlFileInterface.java b/app/src/main/java/com/jens/automation2/XmlFileInterface.java index aa3bddb5..4d48ee93 100644 --- a/app/src/main/java/com/jens/automation2/XmlFileInterface.java +++ b/app/src/main/java/com/jens/automation2/XmlFileInterface.java @@ -217,7 +217,7 @@ public class XmlFileInterface } serializer.endTag(null, "ProfileCollection"); - + serializer.startTag(null, "RuleCollection"); for(int i=0; i + + + + + + + + + + This trigger will be valid if you just disconnected from the wifi specified above OR while the service is still starting and if you\'re not connected to any wifi. If you want the trigger to fire only when you\'re explicitly disconnecting from a certain wifi, add another trigger \"service is not starting\". Class full name by startForegroundService() + Method + GET + POST \ No newline at end of file From 3f76813e80102edfe247425cc385de4e75bd8d17 Mon Sep 17 00:00:00 2001 From: Jens Date: Tue, 12 Dec 2023 23:40:12 +0100 Subject: [PATCH 04/50] Http method selectable --- .../java/com/jens/automation2/Action.java | 2 +- .../java/com/jens/automation2/AsyncTasks.java | 2 +- .../com/jens/automation2/Miscellaneous.java | 40 +++++++++---------- .../main/java/com/jens/automation2/News.java | 2 +- .../metadata/android/en-US/changelogs/138.txt | 3 +- 5 files changed, 24 insertions(+), 25 deletions(-) diff --git a/app/src/main/java/com/jens/automation2/Action.java b/app/src/main/java/com/jens/automation2/Action.java index b6cf8d85..d2694aef 100644 --- a/app/src/main/java/com/jens/automation2/Action.java +++ b/app/src/main/java/com/jens/automation2/Action.java @@ -674,7 +674,7 @@ public class Action Miscellaneous.logEvent("i", "HTTP", "Attempting download of " + url, 4); //getResources().getString("attemptingDownloadOf"); if(this.getParameter1()) // use authentication - new DownloadTask().execute(url, username, password, methodnochindiesignaturaufnehmen); + new DownloadTask().execute(url, username, password, method); else new DownloadTask().execute(url, null, null); } diff --git a/app/src/main/java/com/jens/automation2/AsyncTasks.java b/app/src/main/java/com/jens/automation2/AsyncTasks.java index 793e2f40..4b7401fd 100644 --- a/app/src/main/java/com/jens/automation2/AsyncTasks.java +++ b/app/src/main/java/com/jens/automation2/AsyncTasks.java @@ -22,7 +22,7 @@ public class AsyncTasks try { - String result = Miscellaneous.downloadURL("https://server47.de/automation/?action=getLatestVersionCode", null, null).trim(); + String result = Miscellaneous.downloadURL("https://server47.de/automation/?action=getLatestVersionCode", null, null, ActivityManageActionTriggerUrl.methodGet).trim(); int latestVersion = Integer.parseInt(result); // At this point the update check itself has already been successful. diff --git a/app/src/main/java/com/jens/automation2/Miscellaneous.java b/app/src/main/java/com/jens/automation2/Miscellaneous.java index 97bf03a3..4da9c912 100644 --- a/app/src/main/java/com/jens/automation2/Miscellaneous.java +++ b/app/src/main/java/com/jens/automation2/Miscellaneous.java @@ -48,6 +48,7 @@ import org.apache.http.client.methods.HttpEntityEnclosingRequestBase; import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPost; import org.apache.http.client.methods.HttpPut; +import org.apache.http.client.methods.HttpRequestBase; import org.apache.http.conn.ssl.SSLSocketFactory; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.params.BasicHttpParams; @@ -181,9 +182,7 @@ public class Miscellaneous extends Service public static String downloadURLwithoutCertificateChecking(String url, String username, String password, String method) { - boolean errorFound = false; - - try + try { HttpParams params = new BasicHttpParams(); params.setParameter(HttpProtocolParams.USE_EXPECT_CONTINUE, false); @@ -191,23 +190,23 @@ public class Miscellaneous extends Service HttpClient httpclient = new DefaultHttpClient(params); httpclient = Actions.getInsecureSslClient(httpclient); - HttpEntityEnclosingRequestBase httpRequest; - switch (method) - { - case ActivityManageActionTriggerUrl.methodPost: - httpRequest = new HttpPost(url); - break; - case ActivityManageActionTriggerUrl.methodGet: - default: - httpRequest = new HttpEntityEnclosingRequestBase() + HttpRequestBase httpRequest; + if( + method.equals(ActivityManageActionTriggerUrl.methodPost) + || + (username != null && password != null) + ) + httpRequest = new HttpPost(url); + else + httpRequest = new HttpGet(url); + /*httpRequest = new HttpEntityEnclosingRequestBase() + { + @Override + public String getMethod() { - @Override - public String getMethod() - {somethingMissing - return null; - } - }; - } + return "GET"; + } + };*/ // Add http simple authentication if specified if(username != null && password != null) @@ -227,8 +226,7 @@ public class Miscellaneous extends Service catch(Exception e) { Miscellaneous.logEvent("e", "HTTP error", Log.getStackTraceString(e), 3); - errorFound = true; - return "httpError"; + return "httpError"; } // finally // { diff --git a/app/src/main/java/com/jens/automation2/News.java b/app/src/main/java/com/jens/automation2/News.java index 040b8de4..f4a9ca64 100644 --- a/app/src/main/java/com/jens/automation2/News.java +++ b/app/src/main/java/com/jens/automation2/News.java @@ -79,7 +79,7 @@ public class News if (!(new File(filePath)).exists() || Settings.lastNewsPolltime == Settings.default_lastNewsPolltime || now.getTimeInMillis() >= Settings.lastNewsPolltime + (long)(Settings.newsDisplayForXDays * 24 * 60 * 60 * 1000)) { String newsUrl = "https://server47.de/automation/appNews.php"; - newsContent = Miscellaneous.downloadURL(newsUrl, null, null); + newsContent = Miscellaneous.downloadURL(newsUrl, null, null, ActivityManageActionTriggerUrl.methodGet); // Cache content to local storage if(Miscellaneous.writeStringToFile(filePath, newsContent)) diff --git a/fastlane/metadata/android/en-US/changelogs/138.txt b/fastlane/metadata/android/en-US/changelogs/138.txt index ae7eecd7..7c8e11a0 100644 --- a/fastlane/metadata/android/en-US/changelogs/138.txt +++ b/fastlane/metadata/android/en-US/changelogs/138.txt @@ -1,2 +1,3 @@ * Fixed: Overlay permission for start other program action only required if startByActivity() is selected -* Fixed: Broadcast receiver trigger would not trigger anything, but crash \ No newline at end of file +* Fixed: Broadcast receiver trigger would not trigger anything, but crash +* Added: One can now choose between GET and POST when using triggerURL action \ No newline at end of file From ac74b52aed935a6220b08eb05fd15f48e6a5e42f Mon Sep 17 00:00:00 2001 From: Jens Date: Thu, 14 Dec 2023 00:15:59 +0100 Subject: [PATCH 05/50] Accessibility API --- app/src/apkFlavor/AndroidManifest.xml | 7 ++ app/src/fdroidFlavor/AndroidManifest.xml | 7 ++ app/src/googlePlayFlavor/AndroidManifest.xml | 7 ++ .../java/com/jens/automation2/Action.java | 8 ++ .../java/com/jens/automation2/Actions.java | 9 +++ .../jens/automation2/ActivityManageRule.java | 8 ++ .../jens/automation2/ActivityPermissions.java | 76 +++++++++++++++++++ .../automation2/MyAccessibilityService.java | 29 +++++++ app/src/main/res/values/strings.xml | 3 + 9 files changed, 154 insertions(+) create mode 100644 app/src/main/java/com/jens/automation2/MyAccessibilityService.java diff --git a/app/src/apkFlavor/AndroidManifest.xml b/app/src/apkFlavor/AndroidManifest.xml index 0188eca4..6f843ef0 100644 --- a/app/src/apkFlavor/AndroidManifest.xml +++ b/app/src/apkFlavor/AndroidManifest.xml @@ -261,6 +261,13 @@ android:exported="true" /> + + + + + + \ No newline at end of file diff --git a/app/src/fdroidFlavor/AndroidManifest.xml b/app/src/fdroidFlavor/AndroidManifest.xml index 1a7d25e2..af39e626 100644 --- a/app/src/fdroidFlavor/AndroidManifest.xml +++ b/app/src/fdroidFlavor/AndroidManifest.xml @@ -246,6 +246,13 @@ android:exported="true" /> + + + + + + \ No newline at end of file diff --git a/app/src/googlePlayFlavor/AndroidManifest.xml b/app/src/googlePlayFlavor/AndroidManifest.xml index b4426bca..1c67849b 100644 --- a/app/src/googlePlayFlavor/AndroidManifest.xml +++ b/app/src/googlePlayFlavor/AndroidManifest.xml @@ -245,6 +245,13 @@ android:exported="true" /> + + + + + + \ No newline at end of file diff --git a/app/src/main/java/com/jens/automation2/Action.java b/app/src/main/java/com/jens/automation2/Action.java index d2694aef..2c3c3598 100644 --- a/app/src/main/java/com/jens/automation2/Action.java +++ b/app/src/main/java/com/jens/automation2/Action.java @@ -56,6 +56,7 @@ public class Action startPhoneCall, stopPhoneCall, copyToClipboard, + takeScreenshot, sendTextMessage; public String getFullName(Context context) @@ -140,6 +141,9 @@ public class Action return context.getResources().getString(R.string.endPhoneCall); case copyToClipboard: return context.getResources().getString(R.string.copyTextToClipboard); + case takeScreenshot: + return context.getResources().getString(R.string.takeScreenshot); + default: return "Unknown"; } @@ -307,6 +311,10 @@ public class Action break; case copyToClipboard: returnString.append(Miscellaneous.getAnyContext().getResources().getString(R.string.copyTextToClipboard)); + break; + case takeScreenshot: + returnString.append(Miscellaneous.getAnyContext().getResources().getString(R.string.takeScreenshot)); + break; default: returnString.append(action.toString()); } diff --git a/app/src/main/java/com/jens/automation2/Actions.java b/app/src/main/java/com/jens/automation2/Actions.java index 16037258..3021ce05 100644 --- a/app/src/main/java/com/jens/automation2/Actions.java +++ b/app/src/main/java/com/jens/automation2/Actions.java @@ -1,6 +1,7 @@ package com.jens.automation2; import android.Manifest; +import android.accessibilityservice.AccessibilityService; import android.annotation.SuppressLint; import android.annotation.TargetApi; import android.app.NotificationManager; @@ -2339,4 +2340,12 @@ public class Actions clipboard.setPrimaryClip(clip); } } + + public static void takeScreenshot() + { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) + { + MyAccessibilityService.getInstance().performGlobalAction(AccessibilityService.GLOBAL_ACTION_TAKE_SCREENSHOT); + } + } } \ No newline at end of file diff --git a/app/src/main/java/com/jens/automation2/ActivityManageRule.java b/app/src/main/java/com/jens/automation2/ActivityManageRule.java index 32fda94b..b012e1a0 100644 --- a/app/src/main/java/com/jens/automation2/ActivityManageRule.java +++ b/app/src/main/java/com/jens/automation2/ActivityManageRule.java @@ -2126,6 +2126,8 @@ public class ActivityManageRule extends Activity } else if(types[i].toString().equals(Action_Enum.copyToClipboard.toString())) items.add(new Item(typesLong[i].toString(), R.drawable.clipboard)); + else if(types[i].toString().equals(Action_Enum.takeScreenshot.toString())) + items.add(new Item(typesLong[i].toString(), R.drawable.clipboard)); else items.add(new Item(typesLong[i].toString(), R.drawable.placeholder)); } @@ -2352,6 +2354,12 @@ public class ActivityManageRule extends Activity Intent intent = new Intent(ActivityManageRule.this, ActivityManageActionCopyToClipboard.class); startActivityForResult(intent, requestCodeActionCopyTextToClipboardAdd); } + else if(Action.getActionTypesAsArray()[which].toString().equals(Action_Enum.takeScreenshot.toString())) + { + newAction.setAction(Action_Enum.takeScreenshot); + ruleToEdit.getActionSet().add(newAction); + refreshActionList(); + } } }); diff --git a/app/src/main/java/com/jens/automation2/ActivityPermissions.java b/app/src/main/java/com/jens/automation2/ActivityPermissions.java index f18c2238..43a9c2d4 100644 --- a/app/src/main/java/com/jens/automation2/ActivityPermissions.java +++ b/app/src/main/java/com/jens/automation2/ActivityPermissions.java @@ -18,6 +18,7 @@ import android.os.Bundle; import android.os.PowerManager; import android.provider.Settings; import android.text.Html; +import android.text.TextUtils; import android.util.Log; import android.view.View; import android.widget.Button; @@ -51,6 +52,7 @@ public class ActivityPermissions extends Activity private static final int requestCodeForPermissionsBatteryOptimization = 12048; private static final int requestCodeForPermissionNotificationAccessAndroid13 = 12049; private static final int requestCodeForPermissionsManageOverlay = 12050; + private static final int requestCodeForPermissionsAccessibility = 12051; protected String[] specificPermissionsToRequest = null; public static String intentExtraName = "permissionsToBeRequested"; @@ -305,6 +307,10 @@ public class ActivityPermissions extends Activity { return android.provider.Settings.canDrawOverlays(Miscellaneous.getAnyContext()); } + else if(s.equals(Manifest.permission.BIND_ACCESSIBILITY_SERVICE)) + { + return haveAccessibilityAccess(Miscellaneous.getAnyContext()); + } else { int res = context.checkCallingOrSelfPermission(s); @@ -323,11 +329,57 @@ public class ActivityPermissions extends Activity return active; } + public static boolean haveAccessibilityAccess(Context mContext) + { + int accessibilityEnabled = 0; + + final String service = mContext.getPackageName() + "/com.mytest.accessibility.MyAccessibilityService"; + + boolean accessibilityFound = false; + try + { + accessibilityEnabled = Settings.Secure.getInt(mContext.getApplicationContext().getContentResolver(), Settings.Secure.ACCESSIBILITY_ENABLED); + Log.v(TAG, "accessibilityEnabled = " + accessibilityEnabled); + } + catch (Settings.SettingNotFoundException e) + { + Log.e(TAG, "Error finding setting, default accessibility to not found: " + e.getMessage()); + } + TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':'); + + if (accessibilityEnabled == 1) { + String settingValue = Settings.Secure.getString(mContext + .getApplicationContext().getContentResolver(), + Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES); + if (settingValue != null) { + TextUtils.SimpleStringSplitter splitter = mStringColonSplitter; + splitter.setString(settingValue); + while (splitter.hasNext()) { + String accessabilityService = splitter.next(); + + if (accessabilityService.equalsIgnoreCase(service)) { + return true; + } + } + } + } + + return accessibilityFound; + } + public static void requestOverlay() { Intent intent = new Intent(android.provider.Settings.ACTION_MANAGE_OVERLAY_PERMISSION); ActivityPermissions.getInstance().startActivityForResult(intent, requestCodeForPermissionsManageOverlay); } + + public static void requestBindAccessibilityService() + { + Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS); + intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); + ActivityPermissions.getInstance().startActivityForResult(intent, requestCodeForPermissionsAccessibility); + } + public static void requestDeviceAdmin() { if(!haveDeviceAdmin()) @@ -722,6 +774,8 @@ public class ActivityPermissions extends Activity case stopPhoneCall: addToArrayListUnique(Manifest.permission.ANSWER_PHONE_CALLS, requiredPermissions); break; + case takeScreenshot: + addToArrayListUnique(Manifest.permission.BIND_ACCESSIBILITY_SERVICE, requiredPermissions); default: break; } @@ -971,6 +1025,8 @@ public class ActivityPermissions extends Activity case Manifest.permission.QUERY_ALL_PACKAGES: usingElements.add(getResources().getString(R.string.queryAllPackages)); break; + case Manifest.permission.BIND_ACCESSIBILITY_SERVICE: + usingElements.add(getResources().getString(R.string.bindAccessibilityService)); } return usingElements; @@ -1026,6 +1082,10 @@ public class ActivityPermissions extends Activity if (requestCode == requestCodeForPermissionsManageOverlay) if(havePermission(Manifest.permission.SYSTEM_ALERT_WINDOW, ActivityPermissions.this)) requestPermissions(cachedPermissionsToRequest, true); + + if (requestCode == requestCodeForPermissionsAccessibility) + if(havePermission(Manifest.permission.BIND_ACCESSIBILITY_SERVICE, ActivityPermissions.this)) + requestPermissions(cachedPermissionsToRequest, true); } } @@ -1107,6 +1167,22 @@ public class ActivityPermissions extends Activity diag.show(); return; } + else if (s.equalsIgnoreCase(Manifest.permission.BIND_ACCESSIBILITY_SERVICE)) + { + AlertDialog diag = Miscellaneous.messageBox(getResources().getString(R.string.info), getResources().getString(R.string.accessibilityApiPermissionHint), ActivityPermissions.this); + diag.setOnDismissListener(new DialogInterface.OnDismissListener() + { + @Override + public void onDismiss(DialogInterface dialogInterface) + { + requiredPermissions.remove(s); + cachedPermissionsToRequest = requiredPermissions; + requestBindAccessibilityService(); + } + }); + diag.show(); + return; + } else if (s.equalsIgnoreCase(Manifest.permission.BIND_NOTIFICATION_LISTENER_SERVICE)) { if(Build.VERSION.SDK_INT >= 33) diff --git a/app/src/main/java/com/jens/automation2/MyAccessibilityService.java b/app/src/main/java/com/jens/automation2/MyAccessibilityService.java new file mode 100644 index 00000000..922c91f9 --- /dev/null +++ b/app/src/main/java/com/jens/automation2/MyAccessibilityService.java @@ -0,0 +1,29 @@ +package com.jens.automation2; + +import android.accessibilityservice.AccessibilityService; +import android.view.accessibility.AccessibilityEvent; + +public class MyAccessibilityService extends AccessibilityService +{ + static MyAccessibilityService instance; + + public static MyAccessibilityService getInstance() + { + if(instance == null) + instance = new MyAccessibilityService(); + + return instance; + } + + @Override + public void onAccessibilityEvent(AccessibilityEvent accessibilityEvent) + { + + } + + @Override + public void onInterrupt() + { + + } +} \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index c71f261b..bfaa942c 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -892,4 +892,7 @@ Method GET POST + Take screenshot + Bind to accessibility service + After clicking OK you\'ll be sent to a system dialog. Please select Automation there and allow "Allow accessibility API". \ No newline at end of file From 47898e84eafc709e74f13d88ceecde8714dac699 Mon Sep 17 00:00:00 2001 From: jens Date: Thu, 14 Dec 2023 17:46:55 +0100 Subject: [PATCH 06/50] Comment added --- app/src/main/java/com/jens/automation2/Actions.java | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/app/src/main/java/com/jens/automation2/Actions.java b/app/src/main/java/com/jens/automation2/Actions.java index 16037258..d20eb8e0 100644 --- a/app/src/main/java/com/jens/automation2/Actions.java +++ b/app/src/main/java/com/jens/automation2/Actions.java @@ -2286,6 +2286,13 @@ public class Actions public static void startPhoneCall(Context context, String phoneNumber) { + /* + You can use Intent.ACTION_DIAL instead of Intent.ACTION_CALL. + This shows the dialer with the number already entered, but + allows the user to decide whether to actually make the call or not. + ACTION_DIAL does not require the CALL_PHONE permission. + */ + Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + Uri.encode(phoneNumber))); // intent.setClassName("com.android.phone","com.android.phone.OutgoingCallBroadcaster"); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); From 38644cee28af942352b9a3894d47e4e2a0216528 Mon Sep 17 00:00:00 2001 From: jens Date: Sat, 16 Dec 2023 13:52:18 +0100 Subject: [PATCH 07/50] Take screenshot action added --- app/src/apkFlavor/AndroidManifest.xml | 4 ++ app/src/fdroidFlavor/AndroidManifest.xml | 4 ++ app/src/googlePlayFlavor/AndroidManifest.xml | 4 ++ .../java/com/jens/automation2/Action.java | 3 + .../jens/automation2/ActivityManageRule.java | 2 +- .../jens/automation2/ActivityPermissions.java | 52 ++++++++++++++---- .../automation2/MyAccessibilityService.java | 22 ++++++++ app/src/main/res/drawable-hdpi/copier.png | Bin 0 -> 1771 bytes app/src/main/res/values/strings.xml | 2 + .../res/xml/config_accessibility_service.xml | 12 ++++ .../metadata/android/en-US/changelogs/138.txt | 3 +- 11 files changed, 94 insertions(+), 14 deletions(-) create mode 100644 app/src/main/res/drawable-hdpi/copier.png create mode 100644 app/src/main/res/xml/config_accessibility_service.xml diff --git a/app/src/apkFlavor/AndroidManifest.xml b/app/src/apkFlavor/AndroidManifest.xml index 6f843ef0..41dd23ad 100644 --- a/app/src/apkFlavor/AndroidManifest.xml +++ b/app/src/apkFlavor/AndroidManifest.xml @@ -266,6 +266,10 @@ + + diff --git a/app/src/fdroidFlavor/AndroidManifest.xml b/app/src/fdroidFlavor/AndroidManifest.xml index af39e626..e02e0287 100644 --- a/app/src/fdroidFlavor/AndroidManifest.xml +++ b/app/src/fdroidFlavor/AndroidManifest.xml @@ -251,6 +251,10 @@ + + diff --git a/app/src/googlePlayFlavor/AndroidManifest.xml b/app/src/googlePlayFlavor/AndroidManifest.xml index 1c67849b..cbc088a4 100644 --- a/app/src/googlePlayFlavor/AndroidManifest.xml +++ b/app/src/googlePlayFlavor/AndroidManifest.xml @@ -250,6 +250,10 @@ + + diff --git a/app/src/main/java/com/jens/automation2/Action.java b/app/src/main/java/com/jens/automation2/Action.java index 2c3c3598..da605244 100644 --- a/app/src/main/java/com/jens/automation2/Action.java +++ b/app/src/main/java/com/jens/automation2/Action.java @@ -640,6 +640,9 @@ public class Action case copyToClipboard: Actions.copyToClipboard(context, Miscellaneous.replaceVariablesInText(this.getParameter2(), context)); break; + case takeScreenshot: + Actions.takeScreenshot(); + break; default: Miscellaneous.logEvent("w", "Action", context.getResources().getString(R.string.unknownActionSpecified), 3); break; diff --git a/app/src/main/java/com/jens/automation2/ActivityManageRule.java b/app/src/main/java/com/jens/automation2/ActivityManageRule.java index b012e1a0..b5c62051 100644 --- a/app/src/main/java/com/jens/automation2/ActivityManageRule.java +++ b/app/src/main/java/com/jens/automation2/ActivityManageRule.java @@ -2127,7 +2127,7 @@ public class ActivityManageRule extends Activity else if(types[i].toString().equals(Action_Enum.copyToClipboard.toString())) items.add(new Item(typesLong[i].toString(), R.drawable.clipboard)); else if(types[i].toString().equals(Action_Enum.takeScreenshot.toString())) - items.add(new Item(typesLong[i].toString(), R.drawable.clipboard)); + items.add(new Item(typesLong[i].toString(), R.drawable.copier)); else items.add(new Item(typesLong[i].toString(), R.drawable.placeholder)); } diff --git a/app/src/main/java/com/jens/automation2/ActivityPermissions.java b/app/src/main/java/com/jens/automation2/ActivityPermissions.java index 43a9c2d4..87b47166 100644 --- a/app/src/main/java/com/jens/automation2/ActivityPermissions.java +++ b/app/src/main/java/com/jens/automation2/ActivityPermissions.java @@ -333,31 +333,33 @@ public class ActivityPermissions extends Activity { int accessibilityEnabled = 0; - final String service = mContext.getPackageName() + "/com.mytest.accessibility.MyAccessibilityService"; + final String service = mContext.getPackageName() + "/" + BuildConfig.APPLICATION_ID + ".MyAccessibilityService"; boolean accessibilityFound = false; try { accessibilityEnabled = Settings.Secure.getInt(mContext.getApplicationContext().getContentResolver(), Settings.Secure.ACCESSIBILITY_ENABLED); - Log.v(TAG, "accessibilityEnabled = " + accessibilityEnabled); +// Log.v(TAG, "accessibilityEnabled = " + accessibilityEnabled); } catch (Settings.SettingNotFoundException e) { - Log.e(TAG, "Error finding setting, default accessibility to not found: " + e.getMessage()); +// Log.e(TAG, "Error finding setting, default accessibility to not found: " + e.getMessage()); } TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':'); - if (accessibilityEnabled == 1) { - String settingValue = Settings.Secure.getString(mContext - .getApplicationContext().getContentResolver(), - Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES); - if (settingValue != null) { + if (accessibilityEnabled == 1) + { + String settingValue = Settings.Secure.getString(mContext.getApplicationContext().getContentResolver(), Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES); + if (settingValue != null) + { TextUtils.SimpleStringSplitter splitter = mStringColonSplitter; splitter.setString(settingValue); - while (splitter.hasNext()) { - String accessabilityService = splitter.next(); + while (splitter.hasNext()) + { + String accessibilityService = splitter.next(); - if (accessabilityService.equalsIgnoreCase(service)) { + if (accessibilityService.equalsIgnoreCase(service)) + { return true; } } @@ -1026,7 +1028,9 @@ public class ActivityPermissions extends Activity usingElements.add(getResources().getString(R.string.queryAllPackages)); break; case Manifest.permission.BIND_ACCESSIBILITY_SERVICE: - usingElements.add(getResources().getString(R.string.bindAccessibilityService)); + for(String ruleName : getRulesUsing(Action.Action_Enum.takeScreenshot)) + usingElements.add(String.format(getResources().getString(R.string.ruleXrequiresThis), ruleName)); + break; } return usingElements; @@ -1035,6 +1039,15 @@ public class ActivityPermissions extends Activity @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { + /* + All of the following permissions need to be "manually" activated by the user in some + buried system menu. + In my opinion by mistake the function will be called when the user has just landed + on one of those screens, not when he exits it again. To compensate for that onResume() + is overridden. This enables the permission screen to automatically close after all + required permissions have been granted. + */ + super.onActivityResult(requestCode, resultCode, data); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) @@ -1665,4 +1678,19 @@ public class ActivityPermissions extends Activity return false; } } + + @Override + protected void onResume() + { + super.onResume(); + + for(String p : getRequiredPermissions(false)) + { + if(!havePermission(p, this)) + return; + } + + // have all + setHaveAllPermissions(); + } } \ No newline at end of file diff --git a/app/src/main/java/com/jens/automation2/MyAccessibilityService.java b/app/src/main/java/com/jens/automation2/MyAccessibilityService.java index 922c91f9..79edbd13 100644 --- a/app/src/main/java/com/jens/automation2/MyAccessibilityService.java +++ b/app/src/main/java/com/jens/automation2/MyAccessibilityService.java @@ -1,8 +1,14 @@ package com.jens.automation2; import android.accessibilityservice.AccessibilityService; +import android.os.Build; +import android.util.Log; +import android.view.Display; import android.view.accessibility.AccessibilityEvent; +import androidx.annotation.NonNull; +import androidx.annotation.RequiresApi; + public class MyAccessibilityService extends AccessibilityService { static MyAccessibilityService instance; @@ -10,7 +16,9 @@ public class MyAccessibilityService extends AccessibilityService public static MyAccessibilityService getInstance() { if(instance == null) + { instance = new MyAccessibilityService(); + } return instance; } @@ -26,4 +34,18 @@ public class MyAccessibilityService extends AccessibilityService { } + + @Override + public void onCreate() + { + super.onCreate(); + instance = this; + } + + @Override + protected void onServiceConnected() + { + super.onServiceConnected(); + Miscellaneous.logEvent("i", "Accessibility service", "Service started.", 4); + } } \ No newline at end of file diff --git a/app/src/main/res/drawable-hdpi/copier.png b/app/src/main/res/drawable-hdpi/copier.png new file mode 100644 index 0000000000000000000000000000000000000000..81257cb09d6514515183e80c4bcc6af091dcfc5c GIT binary patch literal 1771 zcmV;P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00009 za7bBm000XT000XT0n*)m`~Uy|2XskIMF-;s4g)3(`?WAh0000QbVXQnQ*UN;cVTj6 z08n9RZgehAMN}YmGcGkQF)%Rna-jeK1}I5HK~!i%?V4>&Q&$*(bxXEnS+f0Dw#>}G z=$zBxoC>;v2%;diAW%ywSSYkwD)L!A6$uOZH<+mTbS~ zN6ezxChDe+k8SeqeUB|uIOX21EF#^TJfVH><(_k%o^yK6y`2I8PBY#SDe;(&S5x{dTI1LoYqYZBk_+8_isDa3dLR`^9E0%=QD1^s2IC;_YWJoJ3t?Gh)C9fCZA z3Q!`%ty&0jpBaO^HVDB@3gK#?i~{7A+;=8qCai(2IeOR0)m!%K zK9r4u26$}{f}0e=m5?u)F=g%2x388Mbuiu(Wolu4#8TL^JrBCN+F1|6;W+WdU@v(A zsooS5jGt<#Af}82*jbP&3Y=e{ATYv?jb56fzBa%B9>SB^VUGu(vo0%nqnB zDIjXuEH*?cO0r@1wjAsQxs0lhHpAYXTfkge2>Z&5p}X4(drJ!-XM+MR_H;cVWkQNU z5epDAG7DL}E@=ME{EaHeRQtiT@6FJ6r`{peB}&NAM}Z-D5u6GSfzwE8z&waHBtxvJ z0%8iwK^Y$l8>3gkL31gj;@n~%IullmT;eemMI>h>gbiM@Hba8c>tT{2;7^)lHLK@A z^U*ZuXv}tqWxEOr(=?!wEe3=Cdyv0kA>;-wpl=sYC}fXVD+O(&KYVJbfTXD9(0-wn zD}|U4Vo=DdNJGu80_g910OoB*hznWBx;%ZlC`gc$*i~?#aywM+$cH`Ka`A5-RO0Zd z!r?Rm;Q4UvA(+I)&D#LiO}gF@!3gXUlPvaxOZpdLtT zAD)cId1_)JBpxIs$rvvixcTEXJUqm>jbfT0yn-CxUz!ht zgZ+5ukrcNUgF-ylt$hE7FWQ?h^^Xv@7K2ihDXJCyH?Cj7x1*&H?kYy2^q>9B!SVvQ z+xs)7{ujfr4W((wWgUL=d}b+y`}gkRAvYI;LZaeA7WLNFR6_T~PPR@l;6khMa5Y=M zxU-n!24&#G+5>xr_b0R@r(JEbG7#DtYT>6F*WGp~c_90A;l6#;W4gLJSVd@eTUMA3 zdbyu_im?aEc^xTKuUc@cyx2IrZ#mO=%%K?SWo4!ecK@WV)&g5n)L_}Y6^b$vVe_U~ zC@;xkt6JIuG}PB}uQB1p*aPLfgy^G}&%IodwI1q^)HoD#^pFMU_59lnp3 zdI9dhS-#k)V|%6yO&Ir90`bg=dZ;oW9%hE1xV;h{At#MD&nU_*n}Jg%UZFPnT~Dz;{ismPsil87EF!pSS|8b1wFDvTIHnF{bncICl7e zOVIOZw;D>cO8&UIjA6ApUb{Ful<4tF)Uw&%^4cH-Hz|aea7GNRaEme%8RyTP<(@-0 zBZeNo$bDu4)+3JB1|hgfA;g3;V#Fdw49!9GfFH|0f&?Sed2JAan-oG!j1;2__k}EN zq)Vs*Jz2x)nmjR`)(c?NMh;=wCS^*!A%MwCPs?ck)O==M6RHV@O9W@Xy1h! z(LbMsw3i?l}Zc4Bgv_S>bsd%bRU0dQ|Y*AOtrlgqUzfj95mBF=`_~ z;rGthSk{}iV!6RGET POST Take screenshot + Bind to accessibility service Bind to accessibility service After clicking OK you\'ll be sent to a system dialog. Please select Automation there and allow "Allow accessibility API". + Required for certains actions. \ No newline at end of file diff --git a/app/src/main/res/xml/config_accessibility_service.xml b/app/src/main/res/xml/config_accessibility_service.xml new file mode 100644 index 00000000..b1f0a7e9 --- /dev/null +++ b/app/src/main/res/xml/config_accessibility_service.xml @@ -0,0 +1,12 @@ + + \ No newline at end of file diff --git a/fastlane/metadata/android/en-US/changelogs/138.txt b/fastlane/metadata/android/en-US/changelogs/138.txt index 7c8e11a0..1ccda421 100644 --- a/fastlane/metadata/android/en-US/changelogs/138.txt +++ b/fastlane/metadata/android/en-US/changelogs/138.txt @@ -1,3 +1,4 @@ * Fixed: Overlay permission for start other program action only required if startByActivity() is selected * Fixed: Broadcast receiver trigger would not trigger anything, but crash -* Added: One can now choose between GET and POST when using triggerURL action \ No newline at end of file +* Added: One can now choose between GET and POST when using triggerURL action +* Added: new action -> take screenshot \ No newline at end of file From c34dfa4af43da547ecee2fe6f8145fe25fad39ac Mon Sep 17 00:00:00 2001 From: Jens Date: Mon, 18 Dec 2023 22:55:18 +0100 Subject: [PATCH 08/50] Bugfix for Android 14 for auto dialing mmi codes --- app/src/main/java/com/jens/automation2/Actions.java | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/app/src/main/java/com/jens/automation2/Actions.java b/app/src/main/java/com/jens/automation2/Actions.java index 3021ce05..52fcd6bc 100644 --- a/app/src/main/java/com/jens/automation2/Actions.java +++ b/app/src/main/java/com/jens/automation2/Actions.java @@ -2287,7 +2287,18 @@ public class Actions public static void startPhoneCall(Context context, String phoneNumber) { - Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + Uri.encode(phoneNumber))); + Intent intent; + + /* + Bug in Android 14 makes it necessary to add double quotes around MMI code. + More precisely it's required for codes containing the # character. + */ + + if(Build.VERSION.SDK_INT >= 34) + intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + Uri.encode("\"" + phoneNumber + "\""))); + else + intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + Uri.encode(phoneNumber))); + // intent.setClassName("com.android.phone","com.android.phone.OutgoingCallBroadcaster"); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.addFlags(Intent.FLAG_FROM_BACKGROUND); From bca8b44ad68dca14efae4118a506e3561cd6fae6 Mon Sep 17 00:00:00 2001 From: Jens Date: Mon, 18 Dec 2023 23:13:18 +0100 Subject: [PATCH 09/50] Bugfix for Android 14 for auto dialing mmi codes --- fastlane/metadata/android/en-US/changelogs/138.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/fastlane/metadata/android/en-US/changelogs/138.txt b/fastlane/metadata/android/en-US/changelogs/138.txt index 1ccda421..b4ffd349 100644 --- a/fastlane/metadata/android/en-US/changelogs/138.txt +++ b/fastlane/metadata/android/en-US/changelogs/138.txt @@ -1,4 +1,5 @@ * Fixed: Overlay permission for start other program action only required if startByActivity() is selected * Fixed: Broadcast receiver trigger would not trigger anything, but crash +* Fixed: Bug in Android 14 (not in Automation!!!) required a change when dialing MMI codes containing a # character. * Added: One can now choose between GET and POST when using triggerURL action * Added: new action -> take screenshot \ No newline at end of file From f641de9893f6b3b90fe314c844c1dcaed26ed56e Mon Sep 17 00:00:00 2001 From: jens Date: Tue, 19 Dec 2023 16:54:10 +0100 Subject: [PATCH 10/50] Take screenshot action added --- .../jens/automation2/ActivityPermissions.java | 65 ++++++++++++------- .../main/res/layout/permissions_activity.xml | 6 ++ app/src/main/res/values-de/strings.xml | 7 ++ app/src/main/res/values-es/strings.xml | 11 +++- app/src/main/res/values-fr/strings.xml | 7 ++ app/src/main/res/values-it/strings.xml | 7 ++ app/src/main/res/values-nl/strings.xml | 7 ++ app/src/main/res/values-pl/strings.xml | 7 ++ app/src/main/res/values-ru/strings.xml | 7 ++ app/src/main/res/values-zh-rCN/strings.xml | 7 ++ app/src/main/res/values/strings.xml | 3 +- 11 files changed, 106 insertions(+), 28 deletions(-) diff --git a/app/src/main/java/com/jens/automation2/ActivityPermissions.java b/app/src/main/java/com/jens/automation2/ActivityPermissions.java index 87b47166..bd04ea0e 100644 --- a/app/src/main/java/com/jens/automation2/ActivityPermissions.java +++ b/app/src/main/java/com/jens/automation2/ActivityPermissions.java @@ -58,7 +58,7 @@ public class ActivityPermissions extends Activity public static String intentExtraName = "permissionsToBeRequested"; Button bCancelPermissions, bRequestPermissions; - TextView tvPermissionsExplanation, tvPermissionsExplanationSystemSettings, tvPermissionsExplanationLong; + TextView tvPermissionsExplanation, tvPermissionsExplanationSystemSettings, tvPermissionsExplanationLong, tvRestrictionPermissionsNotice; static ActivityPermissions instance = null; public final static String permissionNameWireguard = "com.wireguard.android.permission.CONTROL_TUNNELS"; @@ -89,6 +89,7 @@ public class ActivityPermissions extends Activity tvPermissionsExplanation = (TextView)findViewById(R.id.tvPermissionsExplanation); tvPermissionsExplanationSystemSettings = (TextView)findViewById(R.id.tvPermissionsExplanationSystemSettings); tvPermissionsExplanationLong = (TextView)findViewById(R.id.tvPermissionsExplanationLong); + tvRestrictionPermissionsNotice = (TextView)findViewById(R.id.tvRestrictionPermissionsNotice); bCancelPermissions.setOnClickListener(new View.OnClickListener() { @@ -424,10 +425,13 @@ public class ActivityPermissions extends Activity if(!havePermission(Manifest.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS, workingContext)) addToArrayListUnique(Manifest.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS, requiredPermissions); - for(Profile p : Profile.getProfileCollection()) + if(!havePermission(Manifest.permission.READ_EXTERNAL_STORAGE, workingContext)) { - if(p.changeIncomingCallsRingtone || p.changeNotificationRingtone) - addToArrayListUnique(Manifest.permission.READ_EXTERNAL_STORAGE, requiredPermissions); + for (Profile p : Profile.getProfileCollection()) + { + if (p.changeIncomingCallsRingtone || p.changeNotificationRingtone) + addToArrayListUnique(Manifest.permission.READ_EXTERNAL_STORAGE, requiredPermissions); + } } if (!onlyGeneral) @@ -461,27 +465,6 @@ public class ActivityPermissions extends Activity } } } - - /* - Not all permissions need to be asked for. - */ - - /*if(shouldShowRequestPermissionRationale("android.permission.RECORD_AUDIO")) - Toast.makeText(ActivityMainScreen.this, "shouldShowRequestPermissionRationale", Toast.LENGTH_LONG).show(); - else - Toast.makeText(ActivityMainScreen.this, "not shouldShowRequestPermissionRationale", Toast.LENGTH_LONG).show();*/ - -// addToArrayListUnique("Manifest.permission.RECORD_AUDIO", requiredPermissions); - /*int hasPermission = checkSelfPermission(Manifest.permission.RECORD_AUDIO); - if (hasPermission == PackageManager.PERMISSION_DENIED) - { - Toast.makeText(ActivityMainScreen.this, "Don't have record_audio. Requesting...", Toast.LENGTH_LONG).show(); -// requestPermissions(new String[]{"Manifest.permission.CAMERA"}, requestCodeForPermissions); - ActivityCompat.requestPermissions(ActivityMainScreen.this, new String[]{"Manifest.permission.CAMERA"}, requestCodeForPermissions); - } - else - Toast.makeText(ActivityMainScreen.this, "Have record_audio.", Toast.LENGTH_LONG).show();*/ - } return requiredPermissions.toArray(new String[requiredPermissions.size()]); @@ -1158,10 +1141,14 @@ public class ActivityPermissions extends Activity } else if (s.equalsIgnoreCase(Manifest.permission.ACCESS_NOTIFICATION_POLICY)) { + if(BuildConfig.FLAVOR.equals(AutomationService.flavor_name_apk)) + Miscellaneous.messageBox(getResources().getString(R.string.info), getResources().getString(R.string.noticeRestrictedPermissions), ActivityPermissions.this).show(); + requiredPermissions.remove(s); cachedPermissionsToRequest = requiredPermissions; Intent intent = new Intent(android.provider.Settings.ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS); startActivityForResult(intent, requestCodeForPermissionsNotificationPolicy); + return; } else if (s.equalsIgnoreCase(Manifest.permission.SYSTEM_ALERT_WINDOW)) @@ -1684,6 +1671,34 @@ public class ActivityPermissions extends Activity { super.onResume(); + if(Build.VERSION.SDK_INT >= 33 && BuildConfig.FLAVOR.equals(AutomationService.flavor_name_apk)) + { + for (String p : getRequiredPermissions(false)) + { + if (p.equals(Manifest.permission.BIND_NOTIFICATION_LISTENER_SERVICE) || p.equals(Manifest.permission.BIND_ACCESSIBILITY_SERVICE)) + { + tvRestrictionPermissionsNotice.setText(getResources().getString(R.string.noticeRestrictedPermissions)); + + /* + Opening the app's settings directly does not work because the + mentioned 3 dots are only displayed when you went there the hard way. + */ + /* + tvRestrictionPermissionsNotice.setOnClickListener(new View.OnClickListener() + { + @Override + public void onClick(View view) + { + Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS); + intent.setData(Uri.parse("package:" + getPackageName())); + startActivity(intent); + } + })*/; + break; + } + } + } + for(String p : getRequiredPermissions(false)) { if(!havePermission(p, this)) diff --git a/app/src/main/res/layout/permissions_activity.xml b/app/src/main/res/layout/permissions_activity.xml index 83dd4362..2b0567b5 100644 --- a/app/src/main/res/layout/permissions_activity.xml +++ b/app/src/main/res/layout/permissions_activity.xml @@ -66,6 +66,12 @@ + + Dieser Auslöser ist gültig, wenn Sie gerade die Verbindung zu dem oben angegebenen WLAN getrennt haben ODER während der Dienst noch gestartet wird und wenn Sie mit keinem WLAN verbunden sind. Wenn Sie möchten, dass der Auslöser nur ausgelöst wird, wenn Sie die Verbindung zu einem bestimmten WLAN explizit trennen, fügen Sie einen weiteren Auslöser hinzu: \"Der Dienst wird nicht gestartet\". Klassenname per startForegroundService() + Methode + Screenshot erstellen + An den Barrierefreiheitsdienst anbinden + An den Barrierefreiheitsdienst anbinden + Nachdem Sie auf OK geklickt haben, werden Sie zu einem Systemdialog weitergeleitet. Bitte wählen Sie dort Automatisierung aus und erlauben Sie die Barrierefreiheits-API. + Erforderlich für bestimmte Aktionen. + Wenn Sie eine der folgenden Berechtigungen nicht erteilen und eine Systemmeldung wie \"Eingeschränkte Berechtigung\" erhalten, müssen Sie zuerst zu den Android-Einstellungen und dann zu den Anwendungen gehen und Automatisierung auswählen. Nun sollten sich 3 Punkte in der oberen rechten Ecke befinden. Klicken Sie auf \"Eingeschränkte Einstellungen zulassen\". Danach sollte die erforderliche Erlaubnis erteilt werden können. Dies sollte nur für die APK-Version der App gelten, nicht für die von F-Droid oder dem Play Store. \ No newline at end of file diff --git a/app/src/main/res/values-es/strings.xml b/app/src/main/res/values-es/strings.xml index a6852db6..2b1aa293 100644 --- a/app/src/main/res/values-es/strings.xml +++ b/app/src/main/res/values-es/strings.xml @@ -43,7 +43,7 @@ Inserte un nombre. Nombre de usuario Ok - continuar + Continuar Regla Enviar mensajes SMS Leer directorio @@ -197,7 +197,7 @@ Grabar audio Detectar llamadas salientes Detectar el estado del dispositivo - Leer la almacenamiento + Leer el almacenamiento Escribir en el almacenamiento Modificar la configuración del dispositivo Determinar el estado de la bateria @@ -801,4 +801,11 @@ Este activador será válido si acabas de desconectarte del wifi especificado anteriormente O mientras el servicio aún se está iniciando y si no estás conectado a ningún wifi. Si desea que el activador se active solo cuando se desconecte explícitamente de una determinada red WiFi, agregue otro activador \"el servicio no se está iniciando\". Nombre de la clase a través de startForegroundService(( + Método + Tomar captura de pantalla + Enlazar al servicio de accesibilidad + Enlazar al servicio de accesibilidad + Después de hacer clic en Aceptar, se le enviará a un cuadro de diálogo del sistema. Seleccione Automatización allí y permita Permitir API de accesibilidad. + Requerido para ciertas acciones. + Si no le otorga a uno el siguiente permiso y un mensaje del sistema como \"Ajuste restringido\", primero debe ir a la configuración de Android, luego a las aplicaciones, elija Automatización. Ahora debería haber 3 puntos en la esquina superior derecha. Haga clic en \"Permitir configuraciones restringidas\". Después de eso, el permiso necesario debería poder otorgarse. Esto solo debería aplicarse a la versión APK de la aplicación, no a las de F-Droid o Play Store. \ No newline at end of file diff --git a/app/src/main/res/values-fr/strings.xml b/app/src/main/res/values-fr/strings.xml index ba24aefd..59ccb7b9 100644 --- a/app/src/main/res/values-fr/strings.xml +++ b/app/src/main/res/values-fr/strings.xml @@ -801,4 +801,11 @@ Ce déclencheur sera valide si vous venez de vous déconnecter du wifi spécifié ci-dessus OU alors que le service est encore en cours de démarrage et si vous n\'êtes connecté à aucun wifi. Si vous souhaitez que le déclencheur ne se déclenche que lorsque vous vous déconnectez explicitement d\'un certain wifi, ajoutez un autre déclencheur « le service ne démarre pas ». Nom de la classe par startForegroundService() + Méthode + Prendre une capture d\'écran + Se lier au service d\'accessibilité + Se lier au service d\'accessibilité + Après avoir cliqué sur OK, vous serez redirigé vers une boîte de dialogue système. Sélectionnez Automatisation et autorisez l\'option Autoriser l\'API d\'accessibilité. + Obligatoire pour certaines actions. + Si vous ne parvenez pas à accorder à l\'un d\'entre eux l\'autorisation suivante et un message système tel que « Autorisation restreinte », vous devez d\'abord accéder aux paramètres Android, puis aux applications, puis choisir Automatisation. Maintenant, il devrait y avoir 3 points dans le coin supérieur droit. Cliquez sur « Autoriser les paramètres restreints ». Après cela, l\'autorisation nécessaire devrait pouvoir être accordée. Cela ne devrait s\'appliquer qu\'à la version APK de l\'application, pas à celles de F-Droid ou du Play Store. diff --git a/app/src/main/res/values-it/strings.xml b/app/src/main/res/values-it/strings.xml index 2268f498..6fb53289 100644 --- a/app/src/main/res/values-it/strings.xml +++ b/app/src/main/res/values-it/strings.xml @@ -802,4 +802,11 @@ Questo trigger sarà valido se ti sei appena disconnesso dal Wi-Fi specificato sopra OPPURE mentre il servizio è ancora in fase di avvio e se non sei connesso a nessuna rete Wi-Fi. Se vuoi che il trigger si attivi solo quando ti stai disconnettendo esplicitamente da una determinata rete Wi-Fi, aggiungi un altro trigger \"il servizio non si avvia\". Nome della classe di startForegroundService() + Metodo + Fai uno screenshot + Associare al servizio di accessibilità + Associare al servizio di accessibilità + Dopo aver fatto clic su OK, verrà visualizzata una finestra di dialogo di sistema. Seleziona Automazione e consenti Consenti API di accessibilità. + Obbligatorio per determinate azioni. + Se non riesci a concedere una delle seguenti autorizzazioni e un messaggio di sistema come \"Autorizzazione limitata\", devi prima andare alle impostazioni di Android, quindi alle applicazioni, scegli Automazione. Ora dovrebbero esserci 3 punti nell\'angolo in alto a destra. Fai clic su \"Consenti impostazioni limitate\". Dopodiché dovrebbe essere concessa l\'autorizzazione necessaria. Questo dovrebbe valere solo per la versione APK dell\'app, non per quelle di F-Droid o Play Store. diff --git a/app/src/main/res/values-nl/strings.xml b/app/src/main/res/values-nl/strings.xml index 67866c49..f6c296ad 100644 --- a/app/src/main/res/values-nl/strings.xml +++ b/app/src/main/res/values-nl/strings.xml @@ -800,5 +800,12 @@ Deze trigger is geldig als je net de verbinding met de hierboven gespecificeerde wifi hebt verbroken OF terwijl de service nog aan het starten is en als je niet verbonden bent met wifi. Als je wilt dat de trigger alleen wordt geactiveerd wanneer je expliciet de verbinding met een bepaalde wifi verbreekt, voeg dan nog een trigger toe \"service start niet\". Naam van de klasse door startForegroundService() + Methode + Screenshot maken + Binden aan toegankelijkheidsservice + Binden aan toegankelijkheidsservice + Nadat u op OK hebt geklikt, wordt u naar een systeemdialoogvenster gestuurd. Selecteer daar Automatisering en sta Toegankelijkheids-API toestaan toe. + Vereist voor bepaalde acties. + Als u er niet in slaagt om een van de volgende machtigingen en een systeembericht zoals \"Beperkte toestemming\" te verlenen, moet u eerst naar Android-instellingen gaan en vervolgens naar toepassingen en Automatisering kiezen. Nu zouden er 3 stippen in de rechterbovenhoek moeten zijn. Klik op \"Beperkte instellingen toestaan\". Daarna moet de benodigde toestemming aanvaardbaar zijn. Dit zou alleen van toepassing moeten zijn op de APK-versie van de app, niet die van F-Droid of Play Store. diff --git a/app/src/main/res/values-pl/strings.xml b/app/src/main/res/values-pl/strings.xml index 2b524dd4..7903f3bb 100644 --- a/app/src/main/res/values-pl/strings.xml +++ b/app/src/main/res/values-pl/strings.xml @@ -899,4 +899,11 @@ Ten wyzwalacz będzie prawidłowy, jeśli właśnie rozłączyłeś się z Wi-Fi określonym powyżej LUB gdy usługa jest nadal uruchomiona i jeśli nie masz połączenia z żadną siecią Wi-Fi. Jeśli chcesz, aby wyzwalacz uruchamiał się tylko wtedy, gdy jawnie rozłączasz się z określoną siecią Wi-Fi, dodaj kolejny wyzwalacz \"usługa nie uruchamia się\". nazwa klasy przez startForegroundService() + Metoda + Zrób zrzut ekranu + Powiąż z usługą ułatwień dostępu + Powiąż z usługą ułatwień dostępu + Po kliknięciu OK zostaniesz przekierowany do systemowego okna dialogowego. Wybierz tam Automatyzację i zezwól na Zezwalaj na interfejs API ułatwień dostępu. + Wymagane w przypadku niektórych działań. + Jeśli nie uda Ci się przyznać następującego uprawnienia i komunikatu systemowego, takiego jak \"Ograniczone uprawnienia\", musisz najpierw przejść do ustawień Androida, a następnie aplikacji, wybrać Automatyzacja. Teraz w prawym górnym rogu powinny znajdować się 3 kropki. Kliknij \"Zezwól na ustawienia z ograniczeniami\". Następnie powinno być możliwe udzielenie niezbędnego pozwolenia. Powinno to dotyczyć tylko wersji APK aplikacji, a nie tych z F-Droid lub Sklepu Play. diff --git a/app/src/main/res/values-ru/strings.xml b/app/src/main/res/values-ru/strings.xml index 243604ff..3b269bb2 100644 --- a/app/src/main/res/values-ru/strings.xml +++ b/app/src/main/res/values-ru/strings.xml @@ -859,4 +859,11 @@ Этот триггер будет действителен, если вы только что отключились от Wi-Fi, указанного выше, ИЛИ во время запуска службы и если вы не подключены ни к одному Wi-Fi. Если вы хотите, чтобы триггер срабатывал только тогда, когда вы явно отключаетесь от определенного Wi-Fi, добавьте еще один триггер «сервис не запускается». Имя класса no startForegroundService() + Метод + Сделать снимок экрана + Привязка к службе специальных возможностей + Привязка к службе специальных возможностей + После нажатия кнопки «ОК» вы попадете в системное диалоговое окно. Выберите там «Автоматизация» и разрешите «Разрешить API специальных возможностей». + Required for certain actions. + Если вы не можете предоставить одно из следующих разрешений и системное сообщение типа «Ограниченное разрешение», вам нужно сначала перейти в настройки Android, затем в приложения, выбрать «Автоматизация». Теперь в правом верхнем углу должно быть 3 точки. Нажмите «Разрешить ограниченные настройки». После этого необходимо получить необходимое разрешение. Это должно относиться только к APK-версии приложения, а не к версии из F-Droid или Play Store. diff --git a/app/src/main/res/values-zh-rCN/strings.xml b/app/src/main/res/values-zh-rCN/strings.xml index 5c1b9b90..147d5c47 100644 --- a/app/src/main/res/values-zh-rCN/strings.xml +++ b/app/src/main/res/values-zh-rCN/strings.xml @@ -800,4 +800,11 @@ 如果您刚刚断开了与上面指定的 wifi 的连接,或者在服务仍在启动并且您没有连接到任何 wifi,则此触发器将有效。 如果您希望触发器仅在您明确断开与某个 wifi 的连接时触发,请添加另一个触发器\"服务未启动\"。 类名 来自 startForegroundService() + 方法 + 截屏 + 绑定到辅助功能服务 + 绑定到辅助功能服务 + 单击\"确定\"后,您将被发送到系统对话框。请在此处选择\"自动化\",并允许\"允许辅助功能 API\"。 + 对于某些操作是必需的。 + 如果您未能授予以下权限和\"受限权限\"之类的系统消息,则需要先转到 Android 设置,然后转到应用程序,选择自动化。现在右上角应该有 3 个点。单击\"允许受限设置\"。之后,应该可以授予必要的权限。这应该仅适用于应用的 APK 版本,而不适用于 F-Droid 或 Play 商店中的版本。 \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 7651fe98..c94aadb0 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -896,5 +896,6 @@ Bind to accessibility service Bind to accessibility service After clicking OK you\'ll be sent to a system dialog. Please select Automation there and allow "Allow accessibility API". - Required for certains actions. + Required for certain actions. + If you fail to grant one the following permission and a system message like \"Restricted permission\" you need to go to Android settings first, then applications, choose Automation. Now there should be 3 dots in the upper right corner. Click \"Allow restricted settings\". After that the necessary permission should be grantable. This should only apply to the APK version of the app, not the ones from F-Droid or Play Store. \ No newline at end of file From 22191648694e3fa10a4d6bc592483f0138f91145 Mon Sep 17 00:00:00 2001 From: Jens Date: Tue, 19 Dec 2023 23:52:28 +0100 Subject: [PATCH 11/50] Set location service --- app/src/apkFlavor/AndroidManifest.xml | 8 +- app/src/fdroidFlavor/AndroidManifest.xml | 7 +- app/src/googlePlayFlavor/AndroidManifest.xml | 7 +- .../java/com/jens/automation2/Action.java | 14 ++- .../java/com/jens/automation2/Actions.java | 14 +++ .../ActivityManageActionLocationService.java | 64 +++++++++++ .../jens/automation2/ActivityManageRule.java | 40 +++++++ .../jens/automation2/ActivityPermissions.java | 15 ++- ...tivity_manage_action_location_servicei.xml | 100 ++++++++++++++++++ app/src/main/res/values/strings.xml | 8 ++ .../metadata/android/en-US/changelogs/138.txt | 4 +- 11 files changed, 274 insertions(+), 7 deletions(-) create mode 100644 app/src/main/java/com/jens/automation2/ActivityManageActionLocationService.java create mode 100644 app/src/main/res/layout/activity_manage_action_location_servicei.xml diff --git a/app/src/apkFlavor/AndroidManifest.xml b/app/src/apkFlavor/AndroidManifest.xml index 41dd23ad..026b6701 100644 --- a/app/src/apkFlavor/AndroidManifest.xml +++ b/app/src/apkFlavor/AndroidManifest.xml @@ -1,5 +1,7 @@ - + + + - + + - + + = Build.VERSION_CODES.Q) +// Miscellaneous.messageBox(getResources().getString(R.string.app_name), getResources().getString(R.string.android10WifiToggleNotice), ActivityManageActionWifi.this).show(); + + if(getApplicationInfo().targetSdkVersion >= Build.VERSION_CODES.Q) + tvWifiExplanation1.setVisibility(View.VISIBLE); + else + tvWifiExplanation1.setVisibility(View.GONE); + + bActionWifiSave.setOnClickListener(new View.OnClickListener() + { + @Override + public void onClick(View view) + { + Intent response = new Intent(); + response.putExtra(ActivityManageRule.intentNameActionParameter1, rbActionWifiOn.isChecked()); + response.putExtra(ActivityManageRule.intentNameActionParameter2, String.valueOf(chkWifiRunAsRoot.isChecked())); + setResult(RESULT_OK, response); + finish(); + } + }); + } +} diff --git a/app/src/main/java/com/jens/automation2/ActivityManageRule.java b/app/src/main/java/com/jens/automation2/ActivityManageRule.java index b5c62051..f8d5b78d 100644 --- a/app/src/main/java/com/jens/automation2/ActivityManageRule.java +++ b/app/src/main/java/com/jens/automation2/ActivityManageRule.java @@ -141,6 +141,8 @@ public class ActivityManageRule extends Activity final static int requestCodeTriggerCheckVariableEdit = 828; final static int requestCodeActionCopyTextToClipboardAdd = 829; final static int requestCodeActionCopyTextToClipboardEdit = 830; + final static int requestCodeActionSetLocationServiceAdd = 831; + final static int requestCodeActionSetLocationServiceEdit = 832; public static ActivityManageRule getInstance() { @@ -478,6 +480,12 @@ public class ActivityManageRule extends Activity actionCopyToClipboardIntent.putExtra(intentNameActionParameter2, a.getParameter2()); startActivityForResult(actionCopyToClipboardIntent, requestCodeActionCopyTextToClipboardEdit); break; + case setLocationService: + Intent actionSetLocationServiceIntent = new Intent(context, ActivityManageActionLocationService.class); + actionSetLocationServiceIntent.putExtra(intentNameActionParameter1, a.getParameter1()); +// actionSetLocationServiceIntent.putExtra(intentNameActionParameter2, a.getParameter2()); + startActivityForResult(actionSetLocationServiceIntent, requestCodeActionSetLocationServiceEdit); + break; default: Miscellaneous.logEvent("w", "Edit action", "Editing of action type " + a.getAction().toString() + " not implemented, yet.", 4); break; @@ -2047,6 +2055,32 @@ public class ActivityManageRule extends Activity ruleToEdit.getActionSet().get(editIndex).setParameter2(data.getStringExtra(intentNameActionParameter2)); } + this.refreshActionList(); + } + } + else if(requestCode == requestCodeActionSetLocationServiceAdd) + { + if(resultCode == RESULT_OK) + { + newAction.setParentRule(ruleToEdit); + newAction.setParameter1(data.getBooleanExtra(intentNameActionParameter1, false)); +// newAction.setParameter2(data.getStringExtra(intentNameActionParameter2)); + ruleToEdit.getActionSet().add(newAction); + this.refreshActionList(); + } + } + else if(requestCode == requestCodeActionSetLocationServiceEdit) + { + if(resultCode == RESULT_OK) + { + ruleToEdit.getActionSet().get(editIndex).setParentRule(ruleToEdit); + ruleToEdit.getActionSet().get(editIndex).setParameter1(data.getBooleanExtra(intentNameActionParameter1, false)); + +// if(data.hasExtra(intentNameActionParameter2)) +// { +// ruleToEdit.getActionSet().get(editIndex).setParameter2(data.getStringExtra(intentNameActionParameter2)); +// } + this.refreshActionList(); } } @@ -2360,6 +2394,12 @@ public class ActivityManageRule extends Activity ruleToEdit.getActionSet().add(newAction); refreshActionList(); } + else if(Action.getActionTypesAsArray()[which].toString().equals(Action_Enum.setLocationService.toString())) + { + newAction.setAction(Action_Enum.setLocationService); + Intent intent = new Intent(ActivityManageRule.this, ActivityManageActionLocationService.class); + startActivityForResult(intent, requestCodeActionSetLocationServiceAdd); + } } }); diff --git a/app/src/main/java/com/jens/automation2/ActivityPermissions.java b/app/src/main/java/com/jens/automation2/ActivityPermissions.java index bd04ea0e..d3270ddf 100644 --- a/app/src/main/java/com/jens/automation2/ActivityPermissions.java +++ b/app/src/main/java/com/jens/automation2/ActivityPermissions.java @@ -164,7 +164,7 @@ public class ActivityPermissions extends Activity /* Filter location permission and only name it once */ - if(s.equals(Manifest.permission.ACCESS_COARSE_LOCATION) | s.equals(Manifest.permission.ACCESS_FINE_LOCATION)) + if(s.equals(Manifest.permission.ACCESS_COARSE_LOCATION) || s.equals(Manifest.permission.ACCESS_FINE_LOCATION)) { if(!locationPermissionExplained) { @@ -761,6 +761,10 @@ public class ActivityPermissions extends Activity break; case takeScreenshot: addToArrayListUnique(Manifest.permission.BIND_ACCESSIBILITY_SERVICE, requiredPermissions); + break; + case setLocationService: + addToArrayListUnique(Manifest.permission.WRITE_SECURE_SETTINGS, requiredPermissions); + break; default: break; } @@ -1014,6 +1018,10 @@ public class ActivityPermissions extends Activity for(String ruleName : getRulesUsing(Action.Action_Enum.takeScreenshot)) usingElements.add(String.format(getResources().getString(R.string.ruleXrequiresThis), ruleName)); break; + case Manifest.permission.WRITE_SECURE_SETTINGS: + for(String ruleName : getRulesUsing(Action.Action_Enum.setLocationService)) + usingElements.add(String.format(getResources().getString(R.string.ruleXrequiresThis), ruleName)); + break; } return usingElements; @@ -1104,6 +1112,11 @@ public class ActivityPermissions extends Activity permissionList.add(permission); } } + else if(permissionNames.equals(Manifest.permission.WRITE_SECURE_SETTINGS)) + { + Miscellaneous.messageBox(Miscellaneous.getAnyContext().getResources().getString(R.string.info), Miscellaneous.getAnyContext().getResources().getString(R.string.writeSecureSettingsNotice), Miscellaneous.getAnyContext()).show(); + return; + } else { if(!havePermission(permission, Miscellaneous.getAnyContext())) diff --git a/app/src/main/res/layout/activity_manage_action_location_servicei.xml b/app/src/main/res/layout/activity_manage_action_location_servicei.xml new file mode 100644 index 00000000..c31ec514 --- /dev/null +++ b/app/src/main/res/layout/activity_manage_action_location_servicei.xml @@ -0,0 +1,100 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +