mirror of
https://github.com/ImranR98/Obtainium.git
synced 2025-07-13 21:36:42 +02:00
Compare commits
60 Commits
v0.14.22-b
...
v0.14.24-b
Author | SHA1 | Date | |
---|---|---|---|
02f374fdf0 | |||
3536dcd775 | |||
6fa887e536 | |||
71755763b9 | |||
5aabcccfd4 | |||
1e009e2211 | |||
811bc6434b | |||
a5b2d06742 | |||
2c00674835 | |||
9bbff1f436 | |||
c75416f9be | |||
2cb750a7b0 | |||
abfec51964 | |||
e314717a3e | |||
290a78acb5 | |||
b40a4dd243 | |||
b38f6d0581 | |||
5c562a6bdb | |||
238d43f916 | |||
818dc23089 | |||
ddb50940ce | |||
96fb97f4ef | |||
6c0459c2ae | |||
30a9e1ee4b | |||
15a1f1e6e0 | |||
827e2b161c | |||
e95230859e | |||
95cb8ca493 | |||
5735293efb | |||
035df1f338 | |||
36a8b04c4f | |||
3d5411ef28 | |||
df7b74e727 | |||
0138599120 | |||
66446cbffc | |||
ef40f3900f | |||
ce259764a8 | |||
65187cb17b | |||
8f25245ad1 | |||
58b8baa019 | |||
ba1db88c8d | |||
c8f2f42a35 | |||
4e5a9b2af5 | |||
d1ab961c14 | |||
33caf4644e | |||
f34ffe18a8 | |||
4b1c5e111d | |||
dbd7c296c6 | |||
df93cbde8f | |||
d8cbc121a7 | |||
8163cd5c8f | |||
e9e9adb174 | |||
21fdfc1eef | |||
4304251e1e | |||
ae0cd74b0e | |||
6935bff244 | |||
102b9da617 | |||
8bd0d185ae | |||
7bfc5ae0a8 | |||
c72a41db9d |
70
.github/workflows/release.yml
vendored
Normal file
70
.github/workflows/release.yml
vendored
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
name: Build and Release
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: subosito/flutter-action@v2
|
||||||
|
|
||||||
|
- name: Import GPG key
|
||||||
|
id: import_pgp_key
|
||||||
|
uses: crazy-max/ghaction-import-gpg@v6
|
||||||
|
with:
|
||||||
|
gpg_private_key: ${{ secrets.PGP_KEY_BASE64 }}
|
||||||
|
passphrase: ${{ secrets.PGP_PASSPHRASE }}
|
||||||
|
|
||||||
|
- name: Build APKs
|
||||||
|
run: |
|
||||||
|
sed -i 's/signingConfig signingConfigs.release//g' android/app/build.gradle
|
||||||
|
flutter build apk && flutter build apk --split-per-abi
|
||||||
|
rm ./build/app/outputs/flutter-apk/*.sha1
|
||||||
|
ls -l ./build/app/outputs/flutter-apk/
|
||||||
|
|
||||||
|
- name: Sign APKs
|
||||||
|
env:
|
||||||
|
KEYSTORE_BASE64: ${{ secrets.KEYSTORE_BASE64 }}
|
||||||
|
KEYSTORE_PASSWORD: ${{ secrets.KEYSTORE_PASSWORD }}
|
||||||
|
PGP_PASSPHRASE: ${{ secrets.PGP_PASSPHRASE }}
|
||||||
|
run: |
|
||||||
|
echo "${KEYSTORE_BASE64}" | base64 -d > apksign.keystore
|
||||||
|
for apk in ./build/app/outputs/flutter-apk/*-release*.apk; do
|
||||||
|
unsignedFn=${apk/-release/-unsigned}
|
||||||
|
mv "$apk" "$unsignedFn"
|
||||||
|
${ANDROID_HOME}/build-tools/30.0.2/apksigner sign --ks apksign.keystore --ks-pass pass:"${KEYSTORE_PASSWORD}" --out "${apk}" "${unsignedFn}"
|
||||||
|
sha256sum ${apk} | cut -d " " -f 1 > "$apk".sha256
|
||||||
|
gpg --batch --pinentry-mode loopback --passphrase "${PGP_PASSPHRASE}" --sign --detach-sig "$apk".sha256
|
||||||
|
done
|
||||||
|
rm apksign.keystore
|
||||||
|
PGP_KEY_FINGERPRINT="${{ steps.import_pgp_key.outputs.fingerprint }}"
|
||||||
|
|
||||||
|
- name: Extract Version
|
||||||
|
id: extract_version
|
||||||
|
run: |
|
||||||
|
VERSION=$(grep -oP "currentVersion = '\K[^']+" lib/main.dart)
|
||||||
|
echo "version=$VERSION" >> $GITHUB_OUTPUT
|
||||||
|
TAG=$(grep -oP "'.*\\\$currentVersion.*'" lib/main.dart | head -c -2 | tail -c +2 | sed "s/\$currentVersion/$VERSION/g")
|
||||||
|
echo "tag=$TAG" >> $GITHUB_OUTPUT
|
||||||
|
if [ -n "$(echo $TAG | grep -oP '\-beta$')" ]; then BETA=true; else BETA=false; fi
|
||||||
|
echo "beta=$BETA" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Create Tag
|
||||||
|
uses: mathieudutour/github-tag-action@v6.1
|
||||||
|
with:
|
||||||
|
github_token: ${{ secrets.GH_ACCESS_TOKEN }}
|
||||||
|
custom_tag: "${{ steps.extract_version.outputs.tag }}"
|
||||||
|
tag_prefix: ""
|
||||||
|
|
||||||
|
- name: Create Release And Upload APKs
|
||||||
|
uses: ncipollo/release-action@v1
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GH_ACCESS_TOKEN }}
|
||||||
|
tag: "${{ steps.extract_version.outputs.tag }}"
|
||||||
|
prerelease: "${{ steps.extract_version.outputs.beta }}"
|
||||||
|
artifacts: ./build/app/outputs/flutter-apk/*-release*.apk*
|
||||||
|
generateReleaseNotes: true
|
@ -122,15 +122,15 @@
|
|||||||
"ascending": "Aufsteigend",
|
"ascending": "Aufsteigend",
|
||||||
"descending": "Absteigend",
|
"descending": "Absteigend",
|
||||||
"bgUpdateCheckInterval": "Prüfintervall für Hintergrundaktualisierung",
|
"bgUpdateCheckInterval": "Prüfintervall für Hintergrundaktualisierung",
|
||||||
"neverManualOnly": "Nie - nur manuell",
|
"neverManualOnly": "Nie – nur manuell",
|
||||||
"appearance": "Aussehen",
|
"appearance": "Aussehen",
|
||||||
"showWebInAppView": "Quellwebseite in der App-Ansicht anzeigen",
|
"showWebInAppView": "Quellwebseite in der App-Ansicht anzeigen",
|
||||||
"pinUpdates": "Apps mit Aktualisierungen oben anheften",
|
"pinUpdates": "Apps mit Aktualisierungen oben anheften",
|
||||||
"updates": "Aktualisierungen",
|
"updates": "Aktualisierungen",
|
||||||
"sourceSpecific": "Quellenspezifisch",
|
"sourceSpecific": "Quellenspezifisch",
|
||||||
"appSource": "App-Quelle",
|
"appSource": "App-Quelle",
|
||||||
"noLogs": "Keine Protokolle",
|
"noLogs": "Keine Logs",
|
||||||
"appLogs": "App Protokolle",
|
"appLogs": "App-Logs",
|
||||||
"close": "Schließen",
|
"close": "Schließen",
|
||||||
"share": "Teilen",
|
"share": "Teilen",
|
||||||
"appNotFound": "App nicht gefunden",
|
"appNotFound": "App nicht gefunden",
|
||||||
@ -256,25 +256,25 @@
|
|||||||
"filterVersionsByRegEx": "Versionen nach regulären Ausdrücken filtern",
|
"filterVersionsByRegEx": "Versionen nach regulären Ausdrücken filtern",
|
||||||
"trySelectingSuggestedVersionCode": "Versuchen, die vorgeschlagene APK-Code-Version auszuwählen",
|
"trySelectingSuggestedVersionCode": "Versuchen, die vorgeschlagene APK-Code-Version auszuwählen",
|
||||||
"dontSortReleasesList": "Retain release order from API",
|
"dontSortReleasesList": "Retain release order from API",
|
||||||
"reverseSort": "Reverse sorting",
|
"reverseSort": "Umgekehrtes Sortieren",
|
||||||
"debugMenu": "Debug Menu",
|
"debugMenu": "Debug Menü",
|
||||||
"bgTaskStarted": "Background task started - check logs.",
|
"bgTaskStarted": "Hintergrundaufgabe gestartet – Logs prüfen.",
|
||||||
"runBgCheckNow": "Run Background Update Check Now",
|
"runBgCheckNow": "Hintergrundaktualisierungsprüfung jetzt durchführen",
|
||||||
"removeAppQuestion": {
|
"removeAppQuestion": {
|
||||||
"one": "App entfernen?",
|
"one": "App entfernen?",
|
||||||
"other": "Apps entfernen?"
|
"other": "Apps entfernen?"
|
||||||
},
|
},
|
||||||
"tooManyRequestsTryAgainInMinutes": {
|
"tooManyRequestsTryAgainInMinutes": {
|
||||||
"one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut",
|
"one": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minute erneut",
|
||||||
"other": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minuten erneut"
|
"other": "Zu viele Anfragen (Rate begrenzt) – versuchen Sie es in {} Minuten erneut"
|
||||||
},
|
},
|
||||||
"bgUpdateGotErrorRetryInMinutes": {
|
"bgUpdateGotErrorRetryInMinutes": {
|
||||||
"one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant",
|
"one": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minute geplant",
|
||||||
"other": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minuten geplant"
|
"other": "Bei der Aktualisierungsprüfung im Hintergrund wurde ein {} festgestellt, eine erneute Prüfung wird in {} Minuten geplant"
|
||||||
},
|
},
|
||||||
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
"bgCheckFoundUpdatesWillNotifyIfNeeded": {
|
||||||
"one": "Hintergrundaktualisierungsprüfung fand {} Aktualisierung - benachrichtigt den Benutzer, falls erforderlich",
|
"one": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierung – benachrichtigt den Benutzer, falls erforderlich",
|
||||||
"other": "Hintergrundaktualisierungsprüfung fand {} Aktualisierungen - benachrichtigt den Benutzer, falls erforderlich"
|
"other": "Die Hintergrundaktualisierungsprüfung fand {} Aktualisierungen – benachrichtigt den Benutzer, falls erforderlich"
|
||||||
},
|
},
|
||||||
"apps": {
|
"apps": {
|
||||||
"one": "{} App",
|
"one": "{} App",
|
||||||
@ -297,8 +297,8 @@
|
|||||||
"other": "{} Tage"
|
"other": "{} Tage"
|
||||||
},
|
},
|
||||||
"clearedNLogsBeforeXAfterY": {
|
"clearedNLogsBeforeXAfterY": {
|
||||||
"one": "{n} Protokoll gelöscht (vorher = {vorher}, nachher = {nachher})",
|
"one": "{n} Log gelöscht (vorher = {vorher}, nachher = {nachher})",
|
||||||
"other": "{n} Protokolle gelöscht (vorher = {vorher}, nachher = {nachher})"
|
"other": "{n} Logs gelöscht (vorher = {vorher}, nachher = {nachher})"
|
||||||
},
|
},
|
||||||
"xAndNMoreUpdatesAvailable": {
|
"xAndNMoreUpdatesAvailable": {
|
||||||
"one": "{} und 1 weitere App haben Aktualisierungen.",
|
"one": "{} und 1 weitere App haben Aktualisierungen.",
|
||||||
|
@ -11,6 +11,7 @@ class FDroid extends AppSource {
|
|||||||
FDroid() {
|
FDroid() {
|
||||||
host = 'f-droid.org';
|
host = 'f-droid.org';
|
||||||
name = tr('fdroid');
|
name = tr('fdroid');
|
||||||
|
naiveStandardVersionDetection = true;
|
||||||
canSearch = true;
|
canSearch = true;
|
||||||
additionalSourceAppSpecificSettingFormItems = [
|
additionalSourceAppSpecificSettingFormItems = [
|
||||||
[
|
[
|
||||||
|
@ -120,14 +120,14 @@ class HTML extends AppSource {
|
|||||||
GeneratedFormTextField('matchGroupToUse',
|
GeneratedFormTextField('matchGroupToUse',
|
||||||
label: tr('matchGroupToUse'),
|
label: tr('matchGroupToUse'),
|
||||||
required: false,
|
required: false,
|
||||||
hint: '1',
|
hint: '0',
|
||||||
textInputType: const TextInputType.numberWithOptions(),
|
textInputType: const TextInputType.numberWithOptions(),
|
||||||
additionalValidators: [
|
additionalValidators: [
|
||||||
(value) {
|
(value) {
|
||||||
if (value?.isEmpty == true) {
|
if (value?.isEmpty == true) {
|
||||||
value = null;
|
value = null;
|
||||||
}
|
}
|
||||||
value ??= '1';
|
value ??= '0';
|
||||||
return intValidator(value);
|
return intValidator(value);
|
||||||
}
|
}
|
||||||
])
|
])
|
||||||
@ -216,8 +216,12 @@ class HTML extends AppSource {
|
|||||||
if (match.isEmpty) {
|
if (match.isEmpty) {
|
||||||
throw NoVersionError();
|
throw NoVersionError();
|
||||||
}
|
}
|
||||||
version = match.last
|
String matchGroupString =
|
||||||
.group(int.parse(additionalSettings['matchGroupToUse'] as String));
|
(additionalSettings['matchGroupToUse'] as String).trim();
|
||||||
|
if (matchGroupString.isEmpty) {
|
||||||
|
matchGroupString = "0";
|
||||||
|
}
|
||||||
|
version = match.last.group(int.parse(matchGroupString));
|
||||||
if (version?.isEmpty == true) {
|
if (version?.isEmpty == true) {
|
||||||
throw NoVersionError();
|
throw NoVersionError();
|
||||||
}
|
}
|
||||||
|
@ -80,4 +80,20 @@ class Uptodown extends AppSource {
|
|||||||
version, getApkUrlsFromUrls([apkUrl]), AppNames(author, appName),
|
version, getApkUrlsFromUrls([apkUrl]), AppNames(author, appName),
|
||||||
releaseDate: relDate);
|
releaseDate: relDate);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<String> apkUrlPrefetchModifier(
|
||||||
|
String apkUrl, String standardUrl) async {
|
||||||
|
var res = await sourceRequest(apkUrl);
|
||||||
|
if (res.statusCode != 200) {
|
||||||
|
throw getObtainiumHttpError(res);
|
||||||
|
}
|
||||||
|
var html = parse(res.body);
|
||||||
|
var finalUrl =
|
||||||
|
(html.querySelector('.post-download')?.attributes['data-url']);
|
||||||
|
if (finalUrl == null) {
|
||||||
|
throw NoAPKError();
|
||||||
|
}
|
||||||
|
return finalUrl;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,14 +16,13 @@ class WhatsApp extends AppSource {
|
|||||||
@override
|
@override
|
||||||
Future<String> apkUrlPrefetchModifier(
|
Future<String> apkUrlPrefetchModifier(
|
||||||
String apkUrl, String standardUrl) async {
|
String apkUrl, String standardUrl) async {
|
||||||
Response res = await sourceRequest('https://www.whatsapp.com/android');
|
Response res = await sourceRequest('$standardUrl/android');
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
var targetLinks = parse(res.body)
|
var targetLinks = parse(res.body)
|
||||||
.querySelectorAll('a')
|
.querySelectorAll('a')
|
||||||
.map((e) => e.attributes['href'] ?? '')
|
.map((e) => e.attributes['href'] ?? '')
|
||||||
.where((e) => e.isNotEmpty)
|
.where((e) => e.isNotEmpty)
|
||||||
.where((e) =>
|
.where((e) => e.contains('WhatsApp.apk'))
|
||||||
e.contains('content.whatsapp.net') && e.contains('WhatsApp.apk'))
|
|
||||||
.toList();
|
.toList();
|
||||||
if (targetLinks.isEmpty) {
|
if (targetLinks.isEmpty) {
|
||||||
throw NoAPKError();
|
throw NoAPKError();
|
||||||
@ -39,37 +38,16 @@ class WhatsApp extends AppSource {
|
|||||||
String standardUrl,
|
String standardUrl,
|
||||||
Map<String, dynamic> additionalSettings,
|
Map<String, dynamic> additionalSettings,
|
||||||
) async {
|
) async {
|
||||||
Response res = await sourceRequest('https://www.whatsapp.com/android');
|
// This is a CDN link that is consistent per version
|
||||||
if (res.statusCode == 200) {
|
// But it has query params that change constantly
|
||||||
var targetElements = parse(res.body)
|
Uri apkUri =
|
||||||
.querySelectorAll('p')
|
Uri.parse(await apkUrlPrefetchModifier(standardUrl, standardUrl));
|
||||||
.where((element) => element.innerHtml.contains('Version '))
|
var unusableApkUrl = '${apkUri.origin}/${apkUri.path}';
|
||||||
.toList();
|
// So we use the param-less URL is a pseudo-version to add the app and check for updates
|
||||||
if (targetElements.isEmpty) {
|
// See #357 for why we can't scrape the version number directly
|
||||||
throw NoVersionError();
|
// But we re-fetch the URL again with its latest query params at the actual download time
|
||||||
}
|
String version = unusableApkUrl.hashCode.toString();
|
||||||
var vLines = targetElements[0]
|
return APKDetails(version, getApkUrlsFromUrls([unusableApkUrl]),
|
||||||
.innerHtml
|
AppNames('Meta', 'WhatsApp'));
|
||||||
.split('\n')
|
|
||||||
.where((element) => element.contains('Version '))
|
|
||||||
.toList();
|
|
||||||
if (vLines.isEmpty) {
|
|
||||||
throw NoVersionError();
|
|
||||||
}
|
|
||||||
var versionMatch = RegExp('[0-9]+(\\.[0-9]+)+').firstMatch(vLines[0]);
|
|
||||||
if (versionMatch == null) {
|
|
||||||
throw NoVersionError();
|
|
||||||
}
|
|
||||||
String version =
|
|
||||||
vLines[0].substring(versionMatch.start, versionMatch.end);
|
|
||||||
return APKDetails(
|
|
||||||
version,
|
|
||||||
getApkUrlsFromUrls([
|
|
||||||
'https://www.whatsapp.com/android?v=$version&=thisIsaPlaceholder&a=realURLPrefetchedAtDownloadTime'
|
|
||||||
]),
|
|
||||||
AppNames('Meta', 'WhatsApp'));
|
|
||||||
} else {
|
|
||||||
throw getObtainiumHttpError(res);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -65,25 +65,30 @@ class NotImplementedError extends ObtainiumError {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class MultiAppMultiError extends ObtainiumError {
|
class MultiAppMultiError extends ObtainiumError {
|
||||||
Map<String, List<String>> content = {};
|
Map<String, dynamic> rawErrors = {};
|
||||||
|
Map<String, List<String>> idsByErrorString = {};
|
||||||
|
Map<String, String> appIdNames = {};
|
||||||
|
|
||||||
MultiAppMultiError() : super(tr('placeholder'), unexpected: true);
|
MultiAppMultiError() : super(tr('placeholder'), unexpected: true);
|
||||||
|
|
||||||
add(String appId, String string) {
|
add(String appId, dynamic error, {String? appName}) {
|
||||||
var tempIds = content.remove(string);
|
rawErrors[appId] = error;
|
||||||
|
var string = error.toString();
|
||||||
|
var tempIds = idsByErrorString.remove(string);
|
||||||
tempIds ??= [];
|
tempIds ??= [];
|
||||||
tempIds.add(appId);
|
tempIds.add(appId);
|
||||||
content.putIfAbsent(string, () => tempIds!);
|
idsByErrorString.putIfAbsent(string, () => tempIds!);
|
||||||
|
if (appName != null) {
|
||||||
|
appIdNames[appId] = appName;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String errorString(String appId) =>
|
||||||
|
'${appIdNames.containsKey(appId) ? '${appIdNames[appId]} ($appId)' : appId}: ${rawErrors[appId].toString()}';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() {
|
String toString() =>
|
||||||
String finalString = '';
|
idsByErrorString.keys.map((e) => errorString(e)).join('\n\n');
|
||||||
for (var e in content.keys) {
|
|
||||||
finalString += '$e: ${content[e].toString()}\n\n';
|
|
||||||
}
|
|
||||||
return finalString;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
showError(dynamic e, BuildContext context) {
|
showError(dynamic e, BuildContext context) {
|
||||||
|
@ -19,7 +19,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart';
|
|||||||
// ignore: implementation_imports
|
// ignore: implementation_imports
|
||||||
import 'package:easy_localization/src/localization.dart';
|
import 'package:easy_localization/src/localization.dart';
|
||||||
|
|
||||||
const String currentVersion = '0.14.22';
|
const String currentVersion = '0.14.24';
|
||||||
const String currentReleaseTag =
|
const String currentReleaseTag =
|
||||||
'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES
|
'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES
|
||||||
|
|
||||||
|
@ -68,7 +68,7 @@ class AppsPageState extends State<AppsPage> {
|
|||||||
refreshingSince = DateTime.now();
|
refreshingSince = DateTime.now();
|
||||||
});
|
});
|
||||||
return appsProvider.checkUpdates().catchError((e) {
|
return appsProvider.checkUpdates().catchError((e) {
|
||||||
showError(e, context);
|
showError(e is Map ? e['errors'] : e, context);
|
||||||
return <App>[];
|
return <App>[];
|
||||||
}).whenComplete(() {
|
}).whenComplete(() {
|
||||||
setState(() {
|
setState(() {
|
||||||
@ -833,7 +833,7 @@ class AppsPageState extends State<AppsPage> {
|
|||||||
items: const [],
|
items: const [],
|
||||||
initValid: true,
|
initValid: true,
|
||||||
message: tr('installStatusOfXWillBeResetExplanation',
|
message: tr('installStatusOfXWillBeResetExplanation',
|
||||||
args: [plural('app', selectedAppIds.length)]),
|
args: [plural('apps', selectedAppIds.length)]),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
if (values != null) {
|
if (values != null) {
|
||||||
|
@ -217,7 +217,8 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
if (errors.isEmpty) {
|
if (errors.isEmpty) {
|
||||||
// ignore: use_build_context_synchronously
|
// ignore: use_build_context_synchronously
|
||||||
showError(
|
showError(
|
||||||
tr('importedX', args: [plural('app', selectedUrls.length)]),
|
tr('importedX',
|
||||||
|
args: [plural('apps', selectedUrls.length)]),
|
||||||
context);
|
context);
|
||||||
} else {
|
} else {
|
||||||
// ignore: use_build_context_synchronously
|
// ignore: use_build_context_synchronously
|
||||||
@ -274,7 +275,7 @@ class _ImportExportPageState extends State<ImportExportPage> {
|
|||||||
if (errors.isEmpty) {
|
if (errors.isEmpty) {
|
||||||
// ignore: use_build_context_synchronously
|
// ignore: use_build_context_synchronously
|
||||||
showError(
|
showError(
|
||||||
tr('importedX', args: [plural('app', selectedUrls.length)]),
|
tr('importedX', args: [plural('apps', selectedUrls.length)]),
|
||||||
context);
|
context);
|
||||||
} else {
|
} else {
|
||||||
// ignore: use_build_context_synchronously
|
// ignore: use_build_context_synchronously
|
||||||
|
@ -449,7 +449,7 @@ class AppsProvider with ChangeNotifier {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
logs.add(
|
logs.add(
|
||||||
'Could not install APK from XAPK \'${file.path}\': ${e.toString()}');
|
'Could not install APK from XAPK \'${file.path}\': ${e.toString()}');
|
||||||
errors.add(dir.appId, e.toString());
|
errors.add(dir.appId, e, appName: apps[dir.appId]?.name);
|
||||||
}
|
}
|
||||||
} else if (file.path.toLowerCase().endsWith('.obb')) {
|
} else if (file.path.toLowerCase().endsWith('.obb')) {
|
||||||
await moveObbFile(file, dir.appId);
|
await moveObbFile(file, dir.appId);
|
||||||
@ -457,7 +457,7 @@ class AppsProvider with ChangeNotifier {
|
|||||||
}
|
}
|
||||||
if (somethingInstalled) {
|
if (somethingInstalled) {
|
||||||
dir.file.delete(recursive: true);
|
dir.file.delete(recursive: true);
|
||||||
} else if (errors.content.isNotEmpty) {
|
} else if (errors.idsByErrorString.isNotEmpty) {
|
||||||
throw errors;
|
throw errors;
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
@ -677,11 +677,11 @@ class AppsProvider with ChangeNotifier {
|
|||||||
}
|
}
|
||||||
installedIds.add(id);
|
installedIds.add(id);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
errors.add(id, e.toString());
|
errors.add(id, e, appName: apps[id]?.name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (errors.content.isNotEmpty) {
|
if (errors.idsByErrorString.isNotEmpty) {
|
||||||
throw errors;
|
throw errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -738,7 +738,6 @@ class AppsProvider with ChangeNotifier {
|
|||||||
var naiveStandardVersionDetection = SourceProvider()
|
var naiveStandardVersionDetection = SourceProvider()
|
||||||
.getSource(app.url, overrideSource: app.overrideSource)
|
.getSource(app.url, overrideSource: app.overrideSource)
|
||||||
.naiveStandardVersionDetection;
|
.naiveStandardVersionDetection;
|
||||||
;
|
|
||||||
// FIRST, COMPARE THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE ONE IS NULL
|
// FIRST, COMPARE THE APP'S REPORTED AND REAL INSTALLED VERSIONS, WHERE ONE IS NULL
|
||||||
if (installedInfo == null && app.installedVersion != null && !trackOnly) {
|
if (installedInfo == null && app.installedVersion != null && !trackOnly) {
|
||||||
// App says it's installed but isn't really (and isn't track only) - set to not installed
|
// App says it's installed but isn't really (and isn't track only) - set to not installed
|
||||||
@ -1069,7 +1068,8 @@ class AppsProvider with ChangeNotifier {
|
|||||||
|
|
||||||
Future<List<App>> checkUpdates(
|
Future<List<App>> checkUpdates(
|
||||||
{DateTime? ignoreAppsCheckedAfter,
|
{DateTime? ignoreAppsCheckedAfter,
|
||||||
bool throwErrorsForRetry = false}) async {
|
bool throwErrorsForRetry = false,
|
||||||
|
List<String>? specificIds}) async {
|
||||||
List<App> updates = [];
|
List<App> updates = [];
|
||||||
MultiAppMultiError errors = MultiAppMultiError();
|
MultiAppMultiError errors = MultiAppMultiError();
|
||||||
if (!gettingUpdates) {
|
if (!gettingUpdates) {
|
||||||
@ -1077,27 +1077,33 @@ class AppsProvider with ChangeNotifier {
|
|||||||
try {
|
try {
|
||||||
List<String> appIds = getAppsSortedByUpdateCheckTime(
|
List<String> appIds = getAppsSortedByUpdateCheckTime(
|
||||||
ignoreAppsCheckedAfter: ignoreAppsCheckedAfter);
|
ignoreAppsCheckedAfter: ignoreAppsCheckedAfter);
|
||||||
for (int i = 0; i < appIds.length; i++) {
|
if (specificIds != null) {
|
||||||
|
appIds = appIds.where((aId) => specificIds.contains(aId)).toList();
|
||||||
|
}
|
||||||
|
await Future.wait(appIds.map((appId) async {
|
||||||
App? newApp;
|
App? newApp;
|
||||||
try {
|
try {
|
||||||
newApp = await checkUpdate(appIds[i]);
|
newApp = await checkUpdate(appId);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if ((e is RateLimitError || e is SocketException) &&
|
if ((e is RateLimitError || e is SocketException) &&
|
||||||
throwErrorsForRetry) {
|
throwErrorsForRetry) {
|
||||||
rethrow;
|
rethrow;
|
||||||
}
|
}
|
||||||
errors.add(appIds[i], e.toString());
|
errors.add(appId, e, appName: apps[appId]?.name);
|
||||||
}
|
}
|
||||||
if (newApp != null) {
|
if (newApp != null) {
|
||||||
updates.add(newApp);
|
updates.add(newApp);
|
||||||
}
|
}
|
||||||
}
|
}), eagerError: true);
|
||||||
} finally {
|
} finally {
|
||||||
gettingUpdates = false;
|
gettingUpdates = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (errors.content.isNotEmpty) {
|
if (errors.idsByErrorString.isNotEmpty) {
|
||||||
throw errors;
|
var res = <String, dynamic>{};
|
||||||
|
res['errors'] = errors;
|
||||||
|
res['updates'] = updates;
|
||||||
|
throw res;
|
||||||
}
|
}
|
||||||
return updates;
|
return updates;
|
||||||
}
|
}
|
||||||
@ -1314,18 +1320,16 @@ class _APKOriginWarningDialogState extends State<APKOriginWarningDialog> {
|
|||||||
|
|
||||||
/// Background updater function
|
/// Background updater function
|
||||||
///
|
///
|
||||||
/// @param List<String>? toCheck: The appIds to check for updates (default to all apps sorted by last update check time)
|
/// @param List<MapEntry<String, int>>? toCheck: The appIds to check for updates (with the number of previous attempts made per appid) (defaults to all apps)
|
||||||
///
|
///
|
||||||
/// @param List<String>? toInstall: The appIds to attempt to update (defaults to an empty array)
|
/// @param List<String>? toInstall: The appIds to attempt to update (defaults to an empty array)
|
||||||
///
|
///
|
||||||
/// @param int? attemptCount: The number of times the function has failed up to this point (defaults to 0)
|
|
||||||
///
|
|
||||||
/// When toCheck is empty, the function is in "install mode" (else it is in "update mode").
|
/// When toCheck is empty, the function is in "install mode" (else it is in "update mode").
|
||||||
/// In update mode, all apps in toCheck are checked for updates.
|
/// In update mode, all apps in toCheck are checked for updates.
|
||||||
/// If an update is available, the appId is either added to toInstall (if a background update is possible) or the user is notified.
|
/// If an update is available, the appId is either added to toInstall (if a background update is possible) or the user is notified.
|
||||||
/// If there is an error, the function tries to continue after a few minutes (duration depends on the error), up to a maximum of 5 tries.
|
/// If there are errors, the task is run again for the remaining apps after a few minutes (duration depends on the errors), up to a maximum of 5 tries for any app.
|
||||||
///
|
///
|
||||||
/// Once all update checks are complete, the function is called again in install mode.
|
/// Once all update checks are complete, the task is run again in install mode.
|
||||||
/// In this mode, all apps in toInstall are downloaded and installed in the background (install result is unknown).
|
/// In this mode, all apps in toInstall are downloaded and installed in the background (install result is unknown).
|
||||||
/// If there is an error, the function tries to continue after a few minutes (duration depends on the error), up to a maximum of 5 tries.
|
/// If there is an error, the function tries to continue after a few minutes (duration depends on the error), up to a maximum of 5 tries.
|
||||||
///
|
///
|
||||||
@ -1372,87 +1376,109 @@ Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async {
|
|||||||
'BG ${installMode ? 'install' : 'update'} task $taskId: Started (${installMode ? toInstall.length : toCheck.length}).');
|
'BG ${installMode ? 'install' : 'update'} task $taskId: Started (${installMode ? toInstall.length : toCheck.length}).');
|
||||||
|
|
||||||
if (!installMode) {
|
if (!installMode) {
|
||||||
// If in update mode...
|
// If in update mode, we check for updates.
|
||||||
var didCompleteChecking = false;
|
// We divide the results into 4 groups:
|
||||||
CheckingUpdatesNotification? notif;
|
// - toNotify - Apps with updates that the user will be notified about (can't be silently installed)
|
||||||
|
// - toRetry - Apps with update check errors that will be retried in a while
|
||||||
|
// - toThrow - Apps with update check errors that the user will be notified about (no retry)
|
||||||
|
// - toInstall - Apps with updates that will be installed silently
|
||||||
|
// After grouping the updates, we take care of toNotify and toThrow first
|
||||||
|
// Then if toRetry is not empty, we schedule another update task to run in a while (toInstall is retained)
|
||||||
|
// If toRetry is empty, we take care of toInstall
|
||||||
|
|
||||||
|
// Init. vars.
|
||||||
|
List<App> updates = [];
|
||||||
|
List<App> toNotify = [];
|
||||||
|
List<MapEntry<String, int>> toRetry = [];
|
||||||
|
var retryAfterXSeconds = 0;
|
||||||
|
List<String> toThrow = [];
|
||||||
var networkRestricted = false;
|
var networkRestricted = false;
|
||||||
if (appsProvider.settingsProvider.bgUpdatesOnWiFiOnly) {
|
if (appsProvider.settingsProvider.bgUpdatesOnWiFiOnly) {
|
||||||
var netResult = await (Connectivity().checkConnectivity());
|
var netResult = await (Connectivity().checkConnectivity());
|
||||||
networkRestricted = (netResult != ConnectivityResult.wifi) &&
|
networkRestricted = (netResult != ConnectivityResult.wifi) &&
|
||||||
(netResult != ConnectivityResult.ethernet);
|
(netResult != ConnectivityResult.ethernet);
|
||||||
}
|
}
|
||||||
// Loop through all updates and check each
|
MultiAppMultiError? errors;
|
||||||
List<App> toNotify = [];
|
CheckingUpdatesNotification notif =
|
||||||
|
CheckingUpdatesNotification(plural('apps', toCheck.length));
|
||||||
|
|
||||||
try {
|
try {
|
||||||
for (int i = 0; i < toCheck.length; i++) {
|
// Check for updates
|
||||||
var appId = toCheck[i].key;
|
notificationsProvider.notify(notif, cancelExisting: true);
|
||||||
var attemptCount = toCheck[i].value + 1;
|
updates = await appsProvider.checkUpdates(
|
||||||
AppInMemory? app = appsProvider.apps[appId];
|
specificIds: toCheck.map((e) => e.key).toList());
|
||||||
if (app?.app.installedVersion != null) {
|
} catch (e) {
|
||||||
try {
|
// If there were errors, group them into toRetry and toThrow
|
||||||
notificationsProvider.notify(
|
if (e is Map) {
|
||||||
notif = CheckingUpdatesNotification(app?.name ?? appId),
|
updates = e['updates'];
|
||||||
cancelExisting: true);
|
errors = e['errors'];
|
||||||
App? newApp = await appsProvider.checkUpdate(appId);
|
errors!.rawErrors.forEach((key, err) {
|
||||||
if (newApp != null) {
|
logs.add(
|
||||||
if (networkRestricted ||
|
'BG update task $taskId: Got error on checking for $key \'${err.toString()}\'.');
|
||||||
!(await appsProvider.canInstallSilently(app!.app))) {
|
var toCheckApp = toCheck.where((element) => element.key == key).first;
|
||||||
toNotify.add(newApp);
|
if (toCheckApp.value < maxAttempts) {
|
||||||
} else {
|
toRetry.add(MapEntry(toCheckApp.key, toCheckApp.value + 1));
|
||||||
toInstall.add(MapEntry(appId, 0));
|
var minRetryIntervalForThisApp = err is RateLimitError
|
||||||
}
|
? (err.remainingMinutes * 60)
|
||||||
}
|
: e is ClientException
|
||||||
if (i == (toCheck.length - 1)) {
|
? (15 * 60)
|
||||||
didCompleteChecking = true;
|
: pow(toCheckApp.value + 1, 2).toInt();
|
||||||
}
|
if (minRetryIntervalForThisApp > retryAfterXSeconds) {
|
||||||
} catch (e) {
|
retryAfterXSeconds = minRetryIntervalForThisApp;
|
||||||
// If you got an error, move the offender to the back of the line (increment their fail count) and schedule another task to continue checking shortly
|
|
||||||
logs.add(
|
|
||||||
'BG update task $taskId: Got error on checking for $appId \'${e.toString()}\'.');
|
|
||||||
if (attemptCount < maxAttempts) {
|
|
||||||
var remainingSeconds = e is RateLimitError
|
|
||||||
? (i == 0 ? (e.remainingMinutes * 60) : (5 * 60))
|
|
||||||
: e is ClientException
|
|
||||||
? (15 * 60)
|
|
||||||
: pow(attemptCount, 2).toInt();
|
|
||||||
logs.add(
|
|
||||||
'BG update task $taskId: Will continue in $remainingSeconds seconds (with $appId moved to the end of the line).');
|
|
||||||
var remainingToCheck = moveStrToEndMapEntryWithCount(
|
|
||||||
toCheck.sublist(i), MapEntry(appId, attemptCount));
|
|
||||||
AndroidAlarmManager.oneShot(Duration(seconds: remainingSeconds),
|
|
||||||
taskId + 1, bgUpdateCheck,
|
|
||||||
params: {
|
|
||||||
'toCheck': remainingToCheck
|
|
||||||
.map(
|
|
||||||
(entry) => {'key': entry.key, 'value': entry.value})
|
|
||||||
.toList(),
|
|
||||||
'toInstall': toInstall
|
|
||||||
.map(
|
|
||||||
(entry) => {'key': entry.key, 'value': entry.value})
|
|
||||||
.toList(),
|
|
||||||
});
|
|
||||||
break;
|
|
||||||
} else {
|
|
||||||
// If the offender has reached its fail limit, notify the user and remove it from the list (task can continue)
|
|
||||||
toCheck.removeAt(i);
|
|
||||||
i--;
|
|
||||||
notificationsProvider
|
|
||||||
.notify(ErrorCheckingUpdatesNotification(e.toString()));
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (notif != null) {
|
|
||||||
notificationsProvider.cancel(notif.id);
|
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
toThrow.add(key);
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
|
} else {
|
||||||
|
// We don't expect to ever get here in any situation so no need to catch
|
||||||
|
logs.add('Fatal error in BG update task: ${e.toString()}');
|
||||||
|
rethrow;
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
if (toNotify.isNotEmpty) {
|
notificationsProvider.cancel(notif.id);
|
||||||
notificationsProvider.notify(UpdateNotification(toNotify));
|
}
|
||||||
|
|
||||||
|
// Group the updates into toNotify and toInstall
|
||||||
|
for (var i = 0; i < updates.length; i++) {
|
||||||
|
if (networkRestricted ||
|
||||||
|
!(await appsProvider.canInstallSilently(updates[i]))) {
|
||||||
|
toNotify.add(updates[i]);
|
||||||
|
} else {
|
||||||
|
toInstall.add(MapEntry(updates[i].id, 0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If you're done checking and found some silently installable updates, schedule another task which will run in install mode
|
|
||||||
if (didCompleteChecking && toInstall.isNotEmpty) {
|
// Send the update notification
|
||||||
|
if (toNotify.isNotEmpty) {
|
||||||
|
notificationsProvider.notify(UpdateNotification(toNotify));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send the error notifications
|
||||||
|
if (toThrow.isNotEmpty) {
|
||||||
|
for (var appId in toThrow) {
|
||||||
|
notificationsProvider.notify(ErrorCheckingUpdatesNotification(
|
||||||
|
errors!.errorString(appId),
|
||||||
|
id: Random().nextInt(10000)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// if there are update checks to retry, schedule a retry task
|
||||||
|
if (toRetry.isNotEmpty) {
|
||||||
|
logs.add(
|
||||||
|
'BG update task $taskId: Will retry in $retryAfterXSeconds seconds.');
|
||||||
|
AndroidAlarmManager.oneShot(
|
||||||
|
Duration(seconds: retryAfterXSeconds), taskId + 1, bgUpdateCheck,
|
||||||
|
params: {
|
||||||
|
'toCheck': toRetry
|
||||||
|
.map((entry) => {'key': entry.key, 'value': entry.value})
|
||||||
|
.toList(),
|
||||||
|
'toInstall': toInstall
|
||||||
|
.map((entry) => {'key': entry.key, 'value': entry.value})
|
||||||
|
.toList(),
|
||||||
|
});
|
||||||
|
} else if (toInstall.isNotEmpty) {
|
||||||
|
// If there are no more update checks, schedule an install task
|
||||||
logs.add(
|
logs.add(
|
||||||
'BG update task $taskId: Done. Scheduling install task to run immediately.');
|
'BG update task $taskId: Done. Scheduling install task to run immediately.');
|
||||||
AndroidAlarmManager.oneShot(
|
AndroidAlarmManager.oneShot(
|
||||||
@ -1463,11 +1489,14 @@ Future<void> bgUpdateCheck(int taskId, Map<String, dynamic>? params) async {
|
|||||||
.map((entry) => {'key': entry.key, 'value': entry.value})
|
.map((entry) => {'key': entry.key, 'value': entry.value})
|
||||||
.toList()
|
.toList()
|
||||||
});
|
});
|
||||||
} else if (didCompleteChecking) {
|
} else {
|
||||||
logs.add('BG install task $taskId: Done.');
|
logs.add('BG install task $taskId: Done.');
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
// If in install mode...
|
|
||||||
|
if (installMode) {
|
||||||
|
// If in install mode, we install silent updates.
|
||||||
|
|
||||||
var didCompleteInstalling = false;
|
var didCompleteInstalling = false;
|
||||||
var tempObtArr = toInstall.where((element) => element.key == obtainiumId);
|
var tempObtArr = toInstall.where((element) => element.key == obtainiumId);
|
||||||
if (tempObtArr.isNotEmpty) {
|
if (tempObtArr.isNotEmpty) {
|
||||||
|
@ -28,6 +28,7 @@ import 'package:obtainium/app_sources/steammobile.dart';
|
|||||||
import 'package:obtainium/app_sources/telegramapp.dart';
|
import 'package:obtainium/app_sources/telegramapp.dart';
|
||||||
import 'package:obtainium/app_sources/uptodown.dart';
|
import 'package:obtainium/app_sources/uptodown.dart';
|
||||||
import 'package:obtainium/app_sources/vlc.dart';
|
import 'package:obtainium/app_sources/vlc.dart';
|
||||||
|
import 'package:obtainium/app_sources/whatsapp.dart';
|
||||||
import 'package:obtainium/components/generated_form.dart';
|
import 'package:obtainium/components/generated_form.dart';
|
||||||
import 'package:obtainium/custom_errors.dart';
|
import 'package:obtainium/custom_errors.dart';
|
||||||
import 'package:obtainium/mass_app_sources/githubstars.dart';
|
import 'package:obtainium/mass_app_sources/githubstars.dart';
|
||||||
@ -557,7 +558,7 @@ class SourceProvider {
|
|||||||
Mullvad(),
|
Mullvad(),
|
||||||
Signal(),
|
Signal(),
|
||||||
VLC(),
|
VLC(),
|
||||||
// WhatsApp(), // As of 2023-03-20 this is unusable as the version on the webpage is months out of date
|
WhatsApp(), // As of 2023-03-20 this is unusable as the version on the webpage is months out of date
|
||||||
TelegramApp(),
|
TelegramApp(),
|
||||||
SteamMobile(),
|
SteamMobile(),
|
||||||
NeutronCode(),
|
NeutronCode(),
|
||||||
|
24
pubspec.lock
24
pubspec.lock
@ -46,10 +46,10 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: archive
|
name: archive
|
||||||
sha256: d4dc11707abb32ef756ab95678c0d6df54003d98277f7c9aeda14c48e7a38c2f
|
sha256: "06a96f1249f38a00435b3b0c9a3246d934d7dbc8183fc7c9e56989860edb99d4"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.4.3"
|
version: "3.4.4"
|
||||||
args:
|
args:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -291,10 +291,10 @@ packages:
|
|||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
name: flutter_local_notifications
|
name: flutter_local_notifications
|
||||||
sha256: "3002092e5b8ce2f86c3361422e52e6db6776c23ee21e0b2f71b892bf4259ef04"
|
sha256: "6d11ea777496061e583623aaf31923f93a9409ef8fcaeeefdd6cd78bf4fe5bb3"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "15.1.1"
|
version: "16.1.0"
|
||||||
flutter_local_notifications_linux:
|
flutter_local_notifications_linux:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -320,10 +320,10 @@ packages:
|
|||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
name: flutter_markdown
|
name: flutter_markdown
|
||||||
sha256: a10979814c5f4ddbe2b6143fba25d927599e21e3ba65b3862995960606fae78f
|
sha256: "8afc9a6aa6d8e8063523192ba837149dbf3d377a37c0b0fc579149a1fbd4a619"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.6.17+3"
|
version: "0.6.18"
|
||||||
flutter_plugin_android_lifecycle:
|
flutter_plugin_android_lifecycle:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -538,10 +538,10 @@ packages:
|
|||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
name: permission_handler
|
name: permission_handler
|
||||||
sha256: ad65ba9af42a3d067203641de3fd9f547ded1410bad3b84400c2b4899faede70
|
sha256: "284a66179cabdf942f838543e10413246f06424d960c92ba95c84439154fcac8"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "11.0.0"
|
version: "11.0.1"
|
||||||
permission_handler_android:
|
permission_handler_android:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -879,10 +879,10 @@ packages:
|
|||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
name: webview_flutter
|
name: webview_flutter
|
||||||
sha256: "053d454c9475546b4382e9498601fb46293cdac9b3ca93f1a738375bc9a1eee4"
|
sha256: c1ab9b81090705c6069197d9fdc1625e587b52b8d70cdde2339d177ad0dbb98e
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "4.3.0"
|
version: "4.4.1"
|
||||||
webview_flutter_android:
|
webview_flutter_android:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@ -903,10 +903,10 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: webview_flutter_wkwebview
|
name: webview_flutter_wkwebview
|
||||||
sha256: "3c7d56ca4b82654ad1f58aeefb8d593a59224f26d6b2bf8feed074361eb34c86"
|
sha256: "30b9af6bdd457b44c08748b9190d23208b5165357cc2eb57914fee1366c42974"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.9.0"
|
version: "3.9.1"
|
||||||
win32:
|
win32:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
@ -17,7 +17,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev
|
|||||||
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
|
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
|
||||||
# In Windows, build-name is used as the major, minor, and patch parts
|
# In Windows, build-name is used as the major, minor, and patch parts
|
||||||
# of the product and file versions while build-number is used as the build suffix.
|
# of the product and file versions while build-number is used as the build suffix.
|
||||||
version: 0.14.22+214 # When changing this, update the tag in main() accordingly
|
version: 0.14.24+216 # When changing this, update the tag in main() accordingly
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
sdk: '>=3.0.0 <4.0.0'
|
sdk: '>=3.0.0 <4.0.0'
|
||||||
@ -38,7 +38,7 @@ dependencies:
|
|||||||
cupertino_icons: ^1.0.5
|
cupertino_icons: ^1.0.5
|
||||||
path_provider: ^2.0.11
|
path_provider: ^2.0.11
|
||||||
flutter_fgbg: ^0.3.0 # Try removing reliance on this
|
flutter_fgbg: ^0.3.0 # Try removing reliance on this
|
||||||
flutter_local_notifications: ^15.1.0+1
|
flutter_local_notifications: ^16.1.0
|
||||||
provider: ^6.0.3
|
provider: ^6.0.3
|
||||||
http: ^1.0.0
|
http: ^1.0.0
|
||||||
webview_flutter: ^4.0.0
|
webview_flutter: ^4.0.0
|
||||||
|
Reference in New Issue
Block a user