mirror of
				https://github.com/ImranR98/Obtainium.git
				synced 2025-11-03 23:03:29 +01:00 
			
		
		
		
	Compare commits
	
		
			18 Commits
		
	
	
		
			v0.11.31-b
			...
			v0.11.33-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					0213b542e3 | ||
| 
						 | 
					b0e8a4a297 | ||
| 
						 | 
					e72b33ebf2 | ||
| 
						 | 
					283722319b | ||
| 
						 | 
					b406bb5c6a | ||
| 
						 | 
					de2b7fa7a1 | ||
| 
						 | 
					be61220af4 | ||
| 
						 | 
					3e732a4317 | ||
| 
						 | 
					9f2db4e4e7 | ||
| 
						 | 
					78141998f4 | ||
| 
						 | 
					934f237e34 | ||
| 
						 | 
					1b2a9a39e3 | ||
| 
						 | 
					dc52fb6181 | ||
| 
						 | 
					9e4ac397d8 | ||
| 
						 | 
					0ec944eae9 | ||
| 
						 | 
					ad250c30e4 | ||
| 
						 | 
					1090f15508 | ||
| 
						 | 
					666941350e | 
@@ -226,7 +226,7 @@
 | 
			
		||||
    "autoApkFilterByArch": "Nach Möglichkeit versuchen, APKs nach CPU-Architektur zu filtern",
 | 
			
		||||
    "removeAppQuestion": {
 | 
			
		||||
        "one": "App entfernen?",
 | 
			
		||||
        "other": "App entfernen?"
 | 
			
		||||
        "other": "Apps entfernen?"
 | 
			
		||||
    },
 | 
			
		||||
    "tooManyRequestsTryAgainInMinutes": {
 | 
			
		||||
        "one": "Zu viele Anfragen (Rate begrenzt) - versuchen Sie es in {} Minute erneut",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,7 @@
 | 
			
		||||
import 'dart:convert';
 | 
			
		||||
import 'package:easy_localization/easy_localization.dart';
 | 
			
		||||
import 'package:http/http.dart';
 | 
			
		||||
import 'package:obtainium/app_sources/github.dart';
 | 
			
		||||
import 'package:obtainium/components/generated_form.dart';
 | 
			
		||||
import 'package:obtainium/custom_errors.dart';
 | 
			
		||||
import 'package:obtainium/providers/source_provider.dart';
 | 
			
		||||
@@ -35,6 +36,8 @@ class Codeberg extends AppSource {
 | 
			
		||||
    canSearch = true;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  var gh = GitHub();
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  String standardizeURL(String url) {
 | 
			
		||||
    RegExp standardUrlRegEx = RegExp('^https?://$host/[^/]+/[^/]+');
 | 
			
		||||
@@ -54,80 +57,10 @@ class Codeberg extends AppSource {
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
    Map<String, dynamic> additionalSettings,
 | 
			
		||||
  ) async {
 | 
			
		||||
    bool includePrereleases = additionalSettings['includePrereleases'] == true;
 | 
			
		||||
    bool fallbackToOlderReleases =
 | 
			
		||||
        additionalSettings['fallbackToOlderReleases'] == true;
 | 
			
		||||
    String? regexFilter =
 | 
			
		||||
        (additionalSettings['filterReleaseTitlesByRegEx'] as String?)
 | 
			
		||||
                    ?.isNotEmpty ==
 | 
			
		||||
                true
 | 
			
		||||
            ? additionalSettings['filterReleaseTitlesByRegEx']
 | 
			
		||||
            : null;
 | 
			
		||||
    Response res = await get(Uri.parse(
 | 
			
		||||
        'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100'));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      var releases = jsonDecode(res.body) as List<dynamic>;
 | 
			
		||||
 | 
			
		||||
      List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) =>
 | 
			
		||||
          (release['assets'] as List<dynamic>?)
 | 
			
		||||
              ?.map((e) {
 | 
			
		||||
                return e['name'] != null && e['browser_download_url'] != null
 | 
			
		||||
                    ? MapEntry(e['name'] as String,
 | 
			
		||||
                        e['browser_download_url'] as String)
 | 
			
		||||
                    : const MapEntry('', '');
 | 
			
		||||
              })
 | 
			
		||||
              .where((element) => element.key.toLowerCase().endsWith('.apk'))
 | 
			
		||||
              .toList() ??
 | 
			
		||||
          [];
 | 
			
		||||
 | 
			
		||||
      dynamic targetRelease;
 | 
			
		||||
      var prerrelsSkipped = 0;
 | 
			
		||||
      for (int i = 0; i < releases.length; i++) {
 | 
			
		||||
        if (!fallbackToOlderReleases && i > prerrelsSkipped) break;
 | 
			
		||||
        if (!includePrereleases && releases[i]['prerelease'] == true) {
 | 
			
		||||
          prerrelsSkipped++;
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        if (releases[i]['draft'] == true) {
 | 
			
		||||
          // Draft releases not supported
 | 
			
		||||
        }
 | 
			
		||||
        var nameToFilter = releases[i]['name'] as String?;
 | 
			
		||||
        if (nameToFilter == null || nameToFilter.trim().isEmpty) {
 | 
			
		||||
          // Some leave titles empty so tag is used
 | 
			
		||||
          nameToFilter = releases[i]['tag_name'] as String;
 | 
			
		||||
        }
 | 
			
		||||
        if (regexFilter != null &&
 | 
			
		||||
            !RegExp(regexFilter).hasMatch(nameToFilter.trim())) {
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        var apkUrls = getReleaseAPKUrls(releases[i]);
 | 
			
		||||
        if (apkUrls.isEmpty && additionalSettings['trackOnly'] != true) {
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        targetRelease = releases[i];
 | 
			
		||||
        targetRelease['apkUrls'] = apkUrls;
 | 
			
		||||
        break;
 | 
			
		||||
      }
 | 
			
		||||
      if (targetRelease == null) {
 | 
			
		||||
        throw NoReleasesError();
 | 
			
		||||
      }
 | 
			
		||||
      String? version = targetRelease['tag_name'];
 | 
			
		||||
      DateTime? releaseDate = targetRelease['published_at'] != null
 | 
			
		||||
          ? DateTime.parse(targetRelease['published_at'])
 | 
			
		||||
          : null;
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      var changeLog = targetRelease['body'].toString();
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version,
 | 
			
		||||
          targetRelease['apkUrls'] as List<MapEntry<String, String>>,
 | 
			
		||||
          getAppNames(standardUrl),
 | 
			
		||||
          releaseDate: releaseDate,
 | 
			
		||||
          changeLog: changeLog.isEmpty ? null : changeLog);
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
    return gh.getLatestAPKDetailsCommon(
 | 
			
		||||
        'https://$host/api/v1/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100',
 | 
			
		||||
        standardUrl,
 | 
			
		||||
        additionalSettings);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  AppNames getAppNames(String standardUrl) {
 | 
			
		||||
@@ -138,20 +71,9 @@ class Codeberg extends AppSource {
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<Map<String, String>> search(String query) async {
 | 
			
		||||
    Response res = await get(Uri.parse(
 | 
			
		||||
        'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100'));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      Map<String, String> urlsWithDescriptions = {};
 | 
			
		||||
      for (var e in (jsonDecode(res.body)['data'] as List<dynamic>)) {
 | 
			
		||||
        urlsWithDescriptions.addAll({
 | 
			
		||||
          e['html_url'] as String: e['description'] != null
 | 
			
		||||
              ? e['description'] as String
 | 
			
		||||
              : tr('noDescription')
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
      return urlsWithDescriptions;
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
    return gh.searchCommon(
 | 
			
		||||
        query,
 | 
			
		||||
        'https://$host/api/v1/repos/search?q=${Uri.encodeQueryComponent(query)}&limit=100',
 | 
			
		||||
        'data');
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -96,11 +96,9 @@ class GitHub extends AppSource {
 | 
			
		||||
  String? changeLogPageFromStandardUrl(String standardUrl) =>
 | 
			
		||||
      '$standardUrl/releases';
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetails(
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
    Map<String, dynamic> additionalSettings,
 | 
			
		||||
  ) async {
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetailsCommon(String requestUrl,
 | 
			
		||||
      String standardUrl, Map<String, dynamic> additionalSettings,
 | 
			
		||||
      {Function(Response)? onHttpErrorCode}) async {
 | 
			
		||||
    bool includePrereleases = additionalSettings['includePrereleases'] == true;
 | 
			
		||||
    bool fallbackToOlderReleases =
 | 
			
		||||
        additionalSettings['fallbackToOlderReleases'] == true;
 | 
			
		||||
@@ -110,22 +108,40 @@ class GitHub extends AppSource {
 | 
			
		||||
                true
 | 
			
		||||
            ? additionalSettings['filterReleaseTitlesByRegEx']
 | 
			
		||||
            : null;
 | 
			
		||||
    Response res = await get(Uri.parse(
 | 
			
		||||
        'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100'));
 | 
			
		||||
    Response res = await get(Uri.parse(requestUrl));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      var releases = jsonDecode(res.body) as List<dynamic>;
 | 
			
		||||
 | 
			
		||||
      List<String> getReleaseAPKUrls(dynamic release) =>
 | 
			
		||||
      List<MapEntry<String, String>> getReleaseAPKUrls(dynamic release) =>
 | 
			
		||||
          (release['assets'] as List<dynamic>?)
 | 
			
		||||
              ?.map((e) {
 | 
			
		||||
                return e['browser_download_url'] != null
 | 
			
		||||
                    ? e['browser_download_url'] as String
 | 
			
		||||
                    : '';
 | 
			
		||||
                return e['name'] != null && e['browser_download_url'] != null
 | 
			
		||||
                    ? MapEntry(e['name'] as String,
 | 
			
		||||
                        e['browser_download_url'] as String)
 | 
			
		||||
                    : const MapEntry('', '');
 | 
			
		||||
              })
 | 
			
		||||
              .where((element) => element.toLowerCase().endsWith('.apk'))
 | 
			
		||||
              .where((element) => element.key.toLowerCase().endsWith('.apk'))
 | 
			
		||||
              .toList() ??
 | 
			
		||||
          [];
 | 
			
		||||
 | 
			
		||||
      DateTime? getReleaseDateFromRelease(dynamic rel) =>
 | 
			
		||||
          rel?['published_at'] != null
 | 
			
		||||
              ? DateTime.parse(rel['published_at'])
 | 
			
		||||
              : null;
 | 
			
		||||
      releases.sort((a, b) {
 | 
			
		||||
        // See #478
 | 
			
		||||
        if (a == b) {
 | 
			
		||||
          return 0;
 | 
			
		||||
        } else if (a == null) {
 | 
			
		||||
          return -1;
 | 
			
		||||
        } else if (b == null) {
 | 
			
		||||
          return 1;
 | 
			
		||||
        } else {
 | 
			
		||||
          return getReleaseDateFromRelease(a)!
 | 
			
		||||
              .compareTo(getReleaseDateFromRelease(b)!);
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
      releases = releases.reversed.toList();
 | 
			
		||||
      dynamic targetRelease;
 | 
			
		||||
      var prerrelsSkipped = 0;
 | 
			
		||||
      for (int i = 0; i < releases.length; i++) {
 | 
			
		||||
@@ -134,6 +150,10 @@ class GitHub extends AppSource {
 | 
			
		||||
          prerrelsSkipped++;
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        if (releases[i]['draft'] == true) {
 | 
			
		||||
          // Draft releases not supported
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
        var nameToFilter = releases[i]['name'] as String?;
 | 
			
		||||
        if (nameToFilter == null || nameToFilter.trim().isEmpty) {
 | 
			
		||||
          // Some leave titles empty so tag is used
 | 
			
		||||
@@ -155,38 +175,51 @@ class GitHub extends AppSource {
 | 
			
		||||
        throw NoReleasesError();
 | 
			
		||||
      }
 | 
			
		||||
      String? version = targetRelease['tag_name'];
 | 
			
		||||
      DateTime? releaseDate = targetRelease['published_at'] != null
 | 
			
		||||
          ? DateTime.parse(targetRelease['published_at'])
 | 
			
		||||
          : null;
 | 
			
		||||
      DateTime? releaseDate = getReleaseDateFromRelease(targetRelease);
 | 
			
		||||
      if (version == null) {
 | 
			
		||||
        throw NoVersionError();
 | 
			
		||||
      }
 | 
			
		||||
      var changeLog = targetRelease['body'].toString();
 | 
			
		||||
      return APKDetails(
 | 
			
		||||
          version,
 | 
			
		||||
          getApkUrlsFromUrls(targetRelease['apkUrls'] as List<String>),
 | 
			
		||||
          targetRelease['apkUrls'] as List<MapEntry<String, String>>,
 | 
			
		||||
          getAppNames(standardUrl),
 | 
			
		||||
          releaseDate: releaseDate,
 | 
			
		||||
          changeLog: changeLog.isEmpty ? null : changeLog);
 | 
			
		||||
    } else {
 | 
			
		||||
      rateLimitErrorCheck(res);
 | 
			
		||||
      if (onHttpErrorCode != null) {
 | 
			
		||||
        onHttpErrorCode(res);
 | 
			
		||||
      }
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<APKDetails> getLatestAPKDetails(
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
    Map<String, dynamic> additionalSettings,
 | 
			
		||||
  ) async {
 | 
			
		||||
    return getLatestAPKDetailsCommon(
 | 
			
		||||
        'https://${await getCredentialPrefixIfAny()}api.$host/repos${standardUrl.substring('https://$host'.length)}/releases?per_page=100',
 | 
			
		||||
        standardUrl,
 | 
			
		||||
        additionalSettings, onHttpErrorCode: (Response res) {
 | 
			
		||||
      rateLimitErrorCheck(res);
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  AppNames getAppNames(String standardUrl) {
 | 
			
		||||
    String temp = standardUrl.substring(standardUrl.indexOf('://') + 3);
 | 
			
		||||
    List<String> names = temp.substring(temp.indexOf('/') + 1).split('/');
 | 
			
		||||
    return AppNames(names[0], names[1]);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<Map<String, String>> search(String query) async {
 | 
			
		||||
    Response res = await get(Uri.parse(
 | 
			
		||||
        'https://${await getCredentialPrefixIfAny()}api.$host/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100'));
 | 
			
		||||
  Future<Map<String, String>> searchCommon(
 | 
			
		||||
      String query, String requestUrl, String rootProp,
 | 
			
		||||
      {Function(Response)? onHttpErrorCode}) async {
 | 
			
		||||
    Response res = await get(Uri.parse(requestUrl));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      Map<String, String> urlsWithDescriptions = {};
 | 
			
		||||
      for (var e in (jsonDecode(res.body)['items'] as List<dynamic>)) {
 | 
			
		||||
      for (var e in (jsonDecode(res.body)[rootProp] as List<dynamic>)) {
 | 
			
		||||
        urlsWithDescriptions.addAll({
 | 
			
		||||
          e['html_url'] as String:
 | 
			
		||||
              ((e['archived'] == true ? '[ARCHIVED] ' : '') +
 | 
			
		||||
@@ -197,11 +230,23 @@ class GitHub extends AppSource {
 | 
			
		||||
      }
 | 
			
		||||
      return urlsWithDescriptions;
 | 
			
		||||
    } else {
 | 
			
		||||
      rateLimitErrorCheck(res);
 | 
			
		||||
      if (onHttpErrorCode != null) {
 | 
			
		||||
        onHttpErrorCode(res);
 | 
			
		||||
      }
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
  Future<Map<String, String>> search(String query) async {
 | 
			
		||||
    return searchCommon(
 | 
			
		||||
        query,
 | 
			
		||||
        'https://${await getCredentialPrefixIfAny()}api.$host/search/repositories?q=${Uri.encodeQueryComponent(query)}&per_page=100',
 | 
			
		||||
        'items', onHttpErrorCode: (Response res) {
 | 
			
		||||
      rateLimitErrorCheck(res);
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  rateLimitErrorCheck(Response res) {
 | 
			
		||||
    if (res.headers['x-ratelimit-remaining'] == '0') {
 | 
			
		||||
      throw RateLimitError(
 | 
			
		||||
 
 | 
			
		||||
@@ -3,10 +3,19 @@ import 'package:http/http.dart';
 | 
			
		||||
import 'package:obtainium/app_sources/github.dart';
 | 
			
		||||
import 'package:obtainium/custom_errors.dart';
 | 
			
		||||
import 'package:obtainium/providers/source_provider.dart';
 | 
			
		||||
import 'package:obtainium/components/generated_form.dart';
 | 
			
		||||
import 'package:easy_localization/easy_localization.dart';
 | 
			
		||||
 | 
			
		||||
class GitLab extends AppSource {
 | 
			
		||||
  GitLab() {
 | 
			
		||||
    host = 'gitlab.com';
 | 
			
		||||
 | 
			
		||||
    additionalSourceAppSpecificSettingFormItems = [
 | 
			
		||||
      [
 | 
			
		||||
        GeneratedFormSwitch('fallbackToOlderReleases',
 | 
			
		||||
            label: tr('fallbackToOlderReleases'), defaultValue: true)
 | 
			
		||||
      ]
 | 
			
		||||
    ];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @override
 | 
			
		||||
@@ -28,13 +37,15 @@ class GitLab extends AppSource {
 | 
			
		||||
    String standardUrl,
 | 
			
		||||
    Map<String, dynamic> additionalSettings,
 | 
			
		||||
  ) async {
 | 
			
		||||
    bool fallbackToOlderReleases =
 | 
			
		||||
        additionalSettings['fallbackToOlderReleases'] == true;
 | 
			
		||||
    Response res = await get(Uri.parse('$standardUrl/-/tags?format=atom'));
 | 
			
		||||
    if (res.statusCode == 200) {
 | 
			
		||||
      var standardUri = Uri.parse(standardUrl);
 | 
			
		||||
      var parsedHtml = parse(res.body);
 | 
			
		||||
      var entry = parsedHtml.querySelector('entry');
 | 
			
		||||
      var entryContent =
 | 
			
		||||
          parse(parseFragment(entry?.querySelector('content')!.innerHtml).text);
 | 
			
		||||
      var apkDetailsList = parsedHtml.querySelectorAll('entry').map((entry) {
 | 
			
		||||
        var entryContent = parse(
 | 
			
		||||
            parseFragment(entry.querySelector('content')!.innerHtml).text);
 | 
			
		||||
        var apkUrls = [
 | 
			
		||||
          ...getLinksFromParsedHTML(
 | 
			
		||||
              entryContent,
 | 
			
		||||
@@ -51,18 +62,33 @@ class GitLab extends AppSource {
 | 
			
		||||
              .toList()
 | 
			
		||||
        ];
 | 
			
		||||
 | 
			
		||||
      var entryId = entry?.querySelector('id')?.innerHtml;
 | 
			
		||||
        var entryId = entry.querySelector('id')?.innerHtml;
 | 
			
		||||
        var version =
 | 
			
		||||
            entryId == null ? null : Uri.parse(entryId).pathSegments.last;
 | 
			
		||||
      var releaseDateString = entry?.querySelector('updated')?.innerHtml;
 | 
			
		||||
      DateTime? releaseDate =
 | 
			
		||||
          releaseDateString != null ? DateTime.parse(releaseDateString) : null;
 | 
			
		||||
        var releaseDateString = entry.querySelector('updated')?.innerHtml;
 | 
			
		||||
        DateTime? releaseDate = releaseDateString != null
 | 
			
		||||
            ? DateTime.parse(releaseDateString)
 | 
			
		||||
            : null;
 | 
			
		||||
        if (version == null) {
 | 
			
		||||
          throw NoVersionError();
 | 
			
		||||
        }
 | 
			
		||||
        return APKDetails(version, getApkUrlsFromUrls(apkUrls),
 | 
			
		||||
            GitHub().getAppNames(standardUrl),
 | 
			
		||||
            releaseDate: releaseDate);
 | 
			
		||||
      });
 | 
			
		||||
      if (apkDetailsList.isEmpty) {
 | 
			
		||||
        throw NoReleasesError();
 | 
			
		||||
      }
 | 
			
		||||
      if (fallbackToOlderReleases) {
 | 
			
		||||
        if (additionalSettings['trackOnly'] != true) {
 | 
			
		||||
          apkDetailsList =
 | 
			
		||||
              apkDetailsList.where((e) => e.apkUrls.isNotEmpty).toList();
 | 
			
		||||
        }
 | 
			
		||||
        if (apkDetailsList.isEmpty) {
 | 
			
		||||
          throw NoReleasesError();
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      return apkDetailsList.first;
 | 
			
		||||
    } else {
 | 
			
		||||
      throw getObtainiumHttpError(res);
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -31,7 +31,8 @@ class SourceForge extends AppSource {
 | 
			
		||||
      getVersion(String url) {
 | 
			
		||||
        try {
 | 
			
		||||
          var tokens = url.split('/');
 | 
			
		||||
          return tokens[tokens.length - 3];
 | 
			
		||||
          var fi = tokens.indexOf('files');
 | 
			
		||||
          return tokens[tokens[fi + 2] == 'download' ? fi - 1 : fi + 1];
 | 
			
		||||
        } catch (e) {
 | 
			
		||||
          return null;
 | 
			
		||||
        }
 | 
			
		||||
 
 | 
			
		||||
@@ -21,7 +21,7 @@ import 'package:easy_localization/src/easy_localization_controller.dart';
 | 
			
		||||
// ignore: implementation_imports
 | 
			
		||||
import 'package:easy_localization/src/localization.dart';
 | 
			
		||||
 | 
			
		||||
const String currentVersion = '0.11.31';
 | 
			
		||||
const String currentVersion = '0.11.33';
 | 
			
		||||
const String currentReleaseTag =
 | 
			
		||||
    'v$currentVersion-beta'; // KEEP THIS IN SYNC WITH GITHUB RELEASES
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -127,7 +127,8 @@ class _AddAppPageState extends State<AddAppPage> {
 | 
			
		||||
            if (apkUrl == null) {
 | 
			
		||||
              throw ObtainiumError(tr('cancelled'));
 | 
			
		||||
            }
 | 
			
		||||
            app.preferredApkIndex = app.apkUrls.indexOf(apkUrl);
 | 
			
		||||
            app.preferredApkIndex =
 | 
			
		||||
                app.apkUrls.map((e) => e.value).toList().indexOf(apkUrl.value);
 | 
			
		||||
            // ignore: use_build_context_synchronously
 | 
			
		||||
            var downloadedApk = await appsProvider.downloadApp(
 | 
			
		||||
                app, globalNavigatorKey.currentContext);
 | 
			
		||||
 
 | 
			
		||||
@@ -268,9 +268,7 @@ class _AppPageState extends State<AppPage> {
 | 
			
		||||
            }).toList();
 | 
			
		||||
 | 
			
		||||
            return GeneratedFormModal(
 | 
			
		||||
              title: tr('additionalOptions'),
 | 
			
		||||
              items: items,
 | 
			
		||||
            );
 | 
			
		||||
                title: tr('additionalOptions'), items: items);
 | 
			
		||||
          });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -307,6 +305,15 @@ class _AppPageState extends State<AppPage> {
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    getResetInstallStatusButton() => TextButton(
 | 
			
		||||
        onPressed: app?.app == null
 | 
			
		||||
            ? null
 | 
			
		||||
            : () {
 | 
			
		||||
                app!.app.installedVersion = null;
 | 
			
		||||
                appsProvider.saveApps([app.app]);
 | 
			
		||||
              },
 | 
			
		||||
        child: Text(tr('resetInstallStatus')));
 | 
			
		||||
 | 
			
		||||
    getInstallOrUpdateButton() => TextButton(
 | 
			
		||||
        onPressed: (app?.app.installedVersion == null ||
 | 
			
		||||
                    app?.app.installedVersion != app?.app.latestVersion) &&
 | 
			
		||||
@@ -402,7 +409,13 @@ class _AppPageState extends State<AppPage> {
 | 
			
		||||
                            icon: const Icon(Icons.more_horiz),
 | 
			
		||||
                            tooltip: tr('more')),
 | 
			
		||||
                      const SizedBox(width: 16.0),
 | 
			
		||||
                      Expanded(child: getInstallOrUpdateButton()),
 | 
			
		||||
                      Expanded(
 | 
			
		||||
                          child: !isVersionDetectionStandard &&
 | 
			
		||||
                                  app?.app.installedVersion != null &&
 | 
			
		||||
                                  app?.app.installedVersion ==
 | 
			
		||||
                                      app?.app.latestVersion
 | 
			
		||||
                              ? getResetInstallStatusButton()
 | 
			
		||||
                              : getInstallOrUpdateButton()),
 | 
			
		||||
                      const SizedBox(width: 16.0),
 | 
			
		||||
                      Expanded(
 | 
			
		||||
                          child: TextButton(
 | 
			
		||||
 
 | 
			
		||||
@@ -224,6 +224,7 @@ class AppsPageState extends State<AppsPage> {
 | 
			
		||||
            return GeneratedFormModal(
 | 
			
		||||
              title: tr('changes'),
 | 
			
		||||
              items: const [],
 | 
			
		||||
              message: listedApps[index].app.latestVersion,
 | 
			
		||||
              additionalWidgets: [
 | 
			
		||||
                changesUrl != null
 | 
			
		||||
                    ? GestureDetector(
 | 
			
		||||
@@ -405,7 +406,8 @@ class AppsPageState extends State<AppsPage> {
 | 
			
		||||
            children: [
 | 
			
		||||
              Row(mainAxisSize: MainAxisSize.min, children: [
 | 
			
		||||
                Container(
 | 
			
		||||
                    constraints: const BoxConstraints(maxWidth: 150),
 | 
			
		||||
                    constraints: BoxConstraints(
 | 
			
		||||
                        maxWidth: MediaQuery.of(context).size.width / 4),
 | 
			
		||||
                    child: Text(
 | 
			
		||||
                      getVersionText(index),
 | 
			
		||||
                      overflow: TextOverflow.ellipsis,
 | 
			
		||||
@@ -754,8 +756,7 @@ class AppsPageState extends State<AppsPage> {
 | 
			
		||||
      Navigator.of(context).pop();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    resetSelectedAppsInstallStatuses() {
 | 
			
		||||
      () async {
 | 
			
		||||
    resetSelectedAppsInstallStatuses() async {
 | 
			
		||||
      try {
 | 
			
		||||
        var values = await showDialog(
 | 
			
		||||
            context: context,
 | 
			
		||||
@@ -777,7 +778,6 @@ class AppsPageState extends State<AppsPage> {
 | 
			
		||||
      } finally {
 | 
			
		||||
        Navigator.of(context).pop();
 | 
			
		||||
      }
 | 
			
		||||
      };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    showMoreOptionsDialog() {
 | 
			
		||||
@@ -825,7 +825,7 @@ class AppsPageState extends State<AppsPage> {
 | 
			
		||||
                        icon: const Icon(Icons.share),
 | 
			
		||||
                      ),
 | 
			
		||||
                      IconButton(
 | 
			
		||||
                        onPressed: resetSelectedAppsInstallStatuses(),
 | 
			
		||||
                        onPressed: resetSelectedAppsInstallStatuses,
 | 
			
		||||
                        tooltip: tr('resetInstallStatus'),
 | 
			
		||||
                        icon: const Icon(Icons.restore_page_outlined),
 | 
			
		||||
                      ),
 | 
			
		||||
 
 | 
			
		||||
@@ -506,7 +506,7 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> {
 | 
			
		||||
          widget.onlyOneSelectionAllowed ? tr('selectURL') : tr('selectURLs')),
 | 
			
		||||
      content: Column(children: [
 | 
			
		||||
        ...urlWithDescriptionSelections.keys.map((urlWithD) {
 | 
			
		||||
          select(bool? value) {
 | 
			
		||||
          selectThis(bool? value) {
 | 
			
		||||
            setState(() {
 | 
			
		||||
              value ??= false;
 | 
			
		||||
              if (value! && widget.onlyOneSelectionAllowed) {
 | 
			
		||||
@@ -517,11 +517,56 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> {
 | 
			
		||||
            });
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          return Row(children: [
 | 
			
		||||
          var urlLink = GestureDetector(
 | 
			
		||||
              onTap: () {
 | 
			
		||||
                launchUrlString(urlWithD.key,
 | 
			
		||||
                    mode: LaunchMode.externalApplication);
 | 
			
		||||
              },
 | 
			
		||||
              child: Text(
 | 
			
		||||
                Uri.parse(urlWithD.key).path.substring(1),
 | 
			
		||||
                style: const TextStyle(decoration: TextDecoration.underline),
 | 
			
		||||
                textAlign: TextAlign.start,
 | 
			
		||||
              ));
 | 
			
		||||
 | 
			
		||||
          var descriptionText = Text(
 | 
			
		||||
            urlWithD.value.length > 128
 | 
			
		||||
                ? '${urlWithD.value.substring(0, 128)}...'
 | 
			
		||||
                : urlWithD.value,
 | 
			
		||||
            style: const TextStyle(fontStyle: FontStyle.italic, fontSize: 12),
 | 
			
		||||
          );
 | 
			
		||||
 | 
			
		||||
          var selectedUrlsWithDs = urlWithDescriptionSelections.entries
 | 
			
		||||
              .where((e) => e.value)
 | 
			
		||||
              .toList();
 | 
			
		||||
 | 
			
		||||
          var singleSelectTile = ListTile(
 | 
			
		||||
            title: urlLink,
 | 
			
		||||
            subtitle: GestureDetector(
 | 
			
		||||
              onTap: () {
 | 
			
		||||
                setState(() {
 | 
			
		||||
                  selectOnlyOne(urlWithD.key);
 | 
			
		||||
                });
 | 
			
		||||
              },
 | 
			
		||||
              child: descriptionText,
 | 
			
		||||
            ),
 | 
			
		||||
            leading: Radio<String>(
 | 
			
		||||
              value: urlWithD.key,
 | 
			
		||||
              groupValue: selectedUrlsWithDs.isEmpty
 | 
			
		||||
                  ? null
 | 
			
		||||
                  : selectedUrlsWithDs.first.key.key,
 | 
			
		||||
              onChanged: (value) {
 | 
			
		||||
                setState(() {
 | 
			
		||||
                  selectOnlyOne(urlWithD.key);
 | 
			
		||||
                });
 | 
			
		||||
              },
 | 
			
		||||
            ),
 | 
			
		||||
          );
 | 
			
		||||
 | 
			
		||||
          var multiSelectTile = Row(children: [
 | 
			
		||||
            Checkbox(
 | 
			
		||||
                value: urlWithDescriptionSelections[urlWithD],
 | 
			
		||||
                onChanged: (value) {
 | 
			
		||||
                  select(value);
 | 
			
		||||
                  selectThis(value);
 | 
			
		||||
                }),
 | 
			
		||||
            const SizedBox(
 | 
			
		||||
              width: 8,
 | 
			
		||||
@@ -534,28 +579,13 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> {
 | 
			
		||||
                const SizedBox(
 | 
			
		||||
                  height: 8,
 | 
			
		||||
                ),
 | 
			
		||||
                urlLink,
 | 
			
		||||
                GestureDetector(
 | 
			
		||||
                  onTap: () {
 | 
			
		||||
                      launchUrlString(urlWithD.key,
 | 
			
		||||
                          mode: LaunchMode.externalApplication);
 | 
			
		||||
                    selectThis(
 | 
			
		||||
                        !(urlWithDescriptionSelections[urlWithD] ?? false));
 | 
			
		||||
                  },
 | 
			
		||||
                    child: Text(
 | 
			
		||||
                      Uri.parse(urlWithD.key).path.substring(1),
 | 
			
		||||
                      style:
 | 
			
		||||
                          const TextStyle(decoration: TextDecoration.underline),
 | 
			
		||||
                      textAlign: TextAlign.start,
 | 
			
		||||
                    )),
 | 
			
		||||
                GestureDetector(
 | 
			
		||||
                  onTap: () {
 | 
			
		||||
                    select(!(urlWithDescriptionSelections[urlWithD] ?? false));
 | 
			
		||||
                  },
 | 
			
		||||
                  child: Text(
 | 
			
		||||
                    urlWithD.value.length > 128
 | 
			
		||||
                        ? '${urlWithD.value.substring(0, 128)}...'
 | 
			
		||||
                        : urlWithD.value,
 | 
			
		||||
                    style: const TextStyle(
 | 
			
		||||
                        fontStyle: FontStyle.italic, fontSize: 12),
 | 
			
		||||
                  ),
 | 
			
		||||
                  child: descriptionText,
 | 
			
		||||
                ),
 | 
			
		||||
                const SizedBox(
 | 
			
		||||
                  height: 8,
 | 
			
		||||
@@ -563,6 +593,10 @@ class _UrlSelectionModalState extends State<UrlSelectionModal> {
 | 
			
		||||
              ],
 | 
			
		||||
            ))
 | 
			
		||||
          ]);
 | 
			
		||||
 | 
			
		||||
          return widget.onlyOneSelectionAllowed
 | 
			
		||||
              ? singleSelectTile
 | 
			
		||||
              : multiSelectTile;
 | 
			
		||||
        })
 | 
			
		||||
      ]),
 | 
			
		||||
      actions: [
 | 
			
		||||
 
 | 
			
		||||
@@ -304,7 +304,8 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
  Future<MapEntry<String, String>?> confirmApkUrl(
 | 
			
		||||
      App app, BuildContext? context) async {
 | 
			
		||||
    // If the App has more than one APK, the user should pick one (if context provided)
 | 
			
		||||
    MapEntry<String, String>? apkUrl = app.apkUrls[app.preferredApkIndex];
 | 
			
		||||
    MapEntry<String, String>? apkUrl =
 | 
			
		||||
        app.apkUrls[app.preferredApkIndex >= 0 ? app.preferredApkIndex : 0];
 | 
			
		||||
    // get device supported architecture
 | 
			
		||||
    List<String> archs = (await DeviceInfoPlugin().androidInfo).supportedAbis;
 | 
			
		||||
 | 
			
		||||
@@ -365,8 +366,13 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
        apkUrl = await confirmApkUrl(apps[id]!.app, context);
 | 
			
		||||
      }
 | 
			
		||||
      if (apkUrl != null) {
 | 
			
		||||
        int urlInd = apps[id]!.app.apkUrls.indexOf(apkUrl);
 | 
			
		||||
        if (urlInd != apps[id]!.app.preferredApkIndex) {
 | 
			
		||||
        int urlInd = apps[id]!
 | 
			
		||||
            .app
 | 
			
		||||
            .apkUrls
 | 
			
		||||
            .map((e) => e.value)
 | 
			
		||||
            .toList()
 | 
			
		||||
            .indexOf(apkUrl.value);
 | 
			
		||||
        if (urlInd >= 0 && urlInd != apps[id]!.app.preferredApkIndex) {
 | 
			
		||||
          apps[id]!.app.preferredApkIndex = urlInd;
 | 
			
		||||
          await saveApps([apps[id]!.app]);
 | 
			
		||||
        }
 | 
			
		||||
@@ -907,7 +913,7 @@ class AppsProvider with ChangeNotifier {
 | 
			
		||||
 | 
			
		||||
  Future<List<List<String>>> addAppsByURL(List<String> urls) async {
 | 
			
		||||
    List<dynamic> results = await SourceProvider().getAppsByURLNaive(urls,
 | 
			
		||||
        ignoreUrls: apps.values.map((e) => e.app.url).toList());
 | 
			
		||||
        alreadyAddedUrls: apps.values.map((e) => e.app.url).toList());
 | 
			
		||||
    List<App> pps = results[0];
 | 
			
		||||
    Map<String, dynamic> errorsMap = results[1];
 | 
			
		||||
    for (var app in pps) {
 | 
			
		||||
 
 | 
			
		||||
@@ -266,10 +266,12 @@ Map<String, dynamic> getDefaultValuesFromFormItems(
 | 
			
		||||
      .reduce((value, element) => [...value, ...element]));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
getApkUrlsFromUrls(List<String> urls) => urls
 | 
			
		||||
    .map((e) =>
 | 
			
		||||
        MapEntry(e.split('/').where((el) => el.trim().isNotEmpty).last, e))
 | 
			
		||||
    .toList();
 | 
			
		||||
List<MapEntry<String, String>> getApkUrlsFromUrls(List<String> urls) =>
 | 
			
		||||
    urls.map((e) {
 | 
			
		||||
      var segments = e.split('/').where((el) => el.trim().isNotEmpty);
 | 
			
		||||
      var apkSegs = segments.where((s) => s.toLowerCase().endsWith('.apk'));
 | 
			
		||||
      return MapEntry(apkSegs.isNotEmpty ? apkSegs.last : segments.last, e);
 | 
			
		||||
    }).toList();
 | 
			
		||||
 | 
			
		||||
class AppSource {
 | 
			
		||||
  String? host;
 | 
			
		||||
@@ -517,11 +519,14 @@ class SourceProvider {
 | 
			
		||||
 | 
			
		||||
  // Returns errors in [results, errors] instead of throwing them
 | 
			
		||||
  Future<List<dynamic>> getAppsByURLNaive(List<String> urls,
 | 
			
		||||
      {List<String> ignoreUrls = const []}) async {
 | 
			
		||||
      {List<String> alreadyAddedUrls = const []}) async {
 | 
			
		||||
    List<App> apps = [];
 | 
			
		||||
    Map<String, dynamic> errors = {};
 | 
			
		||||
    for (var url in urls.where((element) => !ignoreUrls.contains(element))) {
 | 
			
		||||
    for (var url in urls) {
 | 
			
		||||
      try {
 | 
			
		||||
        if (alreadyAddedUrls.contains(url)) {
 | 
			
		||||
          throw ObtainiumError(tr('appAlreadyAdded'));
 | 
			
		||||
        }
 | 
			
		||||
        var source = getSource(url);
 | 
			
		||||
        apps.add(await getApp(
 | 
			
		||||
            source,
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										44
									
								
								pubspec.lock
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								pubspec.lock
									
									
									
									
									
								
							@@ -181,10 +181,10 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: file_picker
 | 
			
		||||
      sha256: dcde5ad1a0cebcf3715ea3f24d0db1888bf77027a26c77d7779e8ef63b8ade62
 | 
			
		||||
      sha256: b85eb92b175767fdaa0c543bf3b0d1f610fe966412ea72845fe5ba7801e763ff
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "5.2.9"
 | 
			
		||||
    version: "5.2.10"
 | 
			
		||||
  flutter:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description: flutter
 | 
			
		||||
@@ -417,10 +417,10 @@ packages:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: path_provider_android
 | 
			
		||||
      sha256: "019f18c9c10ae370b08dce1f3e3b73bc9f58e7f087bb5e921f06529438ac0ae7"
 | 
			
		||||
      sha256: da97262be945a72270513700a92b39dd2f4a54dad55d061687e2e37a6390366a
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.0.24"
 | 
			
		||||
    version: "2.0.25"
 | 
			
		||||
  path_provider_foundation:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -561,10 +561,10 @@ packages:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: shared_preferences_android
 | 
			
		||||
      sha256: "1bc5d734b109c327b922b0891b41fc51483ccbd53c0d19952b7e230349a5d90b"
 | 
			
		||||
      sha256: "7fa90471a6875d26ad78c7e4a675874b2043874586891128dc5899662c97db46"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.1.1"
 | 
			
		||||
    version: "2.1.2"
 | 
			
		||||
  shared_preferences_foundation:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -622,18 +622,18 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: sqflite
 | 
			
		||||
      sha256: "500d6fec583d2c021f2d25a056d96654f910662c64f836cd2063167b8f1fa758"
 | 
			
		||||
      sha256: e7dfb6482d5d02b661d0b2399efa72b98909e5aa7b8336e1fb37e226264ade00
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.2.6"
 | 
			
		||||
    version: "2.2.7"
 | 
			
		||||
  sqflite_common:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: sqflite_common
 | 
			
		||||
      sha256: "963dad8c4aa2f814ce7d2d5b1da2f36f31bd1a439d8f27e3dc189bb9d26bc684"
 | 
			
		||||
      sha256: "220831bf0bd5333ff2445eee35ec131553b804e6b5d47a4a37ca6f5eb66e282c"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.4.3"
 | 
			
		||||
    version: "2.4.4"
 | 
			
		||||
  stack_trace:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -662,10 +662,10 @@ packages:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: synchronized
 | 
			
		||||
      sha256: "33b31b6beb98100bf9add464a36a8dd03eb10c7a8cf15aeec535e9b054aaf04b"
 | 
			
		||||
      sha256: "5fcbd27688af6082f5abd611af56ee575342c30e87541d0245f7ff99faa02c60"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.0.1"
 | 
			
		||||
    version: "3.1.0"
 | 
			
		||||
  term_glyph:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -686,10 +686,10 @@ packages:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: timezone
 | 
			
		||||
      sha256: "24c8fcdd49a805d95777a39064862133ff816ebfffe0ceff110fb5960e557964"
 | 
			
		||||
      sha256: "1cfd8ddc2d1cfd836bc93e67b9be88c3adaeca6f40a00ca999104c30693cdca0"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "0.9.1"
 | 
			
		||||
    version: "0.9.2"
 | 
			
		||||
  typed_data:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
@@ -782,34 +782,34 @@ packages:
 | 
			
		||||
    dependency: "direct main"
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter
 | 
			
		||||
      sha256: "47663d51a9061451aa3880a214ee9a65dcbb933b77bc44388e194279ab3ccaf6"
 | 
			
		||||
      sha256: "1a37bdbaaf5fbe09ad8579ab09ecfd473284ce482f900b5aea27cf834386a567"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "4.0.7"
 | 
			
		||||
    version: "4.2.0"
 | 
			
		||||
  webview_flutter_android:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter_android
 | 
			
		||||
      sha256: bdd3ddbeaf341c75620e153d22b6f4f6c4a342fe4439ed3a10db74dd82e65d1c
 | 
			
		||||
      sha256: "134ed5d36127b6f5865e86a82174886eae0b983dacd8df14b0448371debde755"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.5.1"
 | 
			
		||||
    version: "3.6.0"
 | 
			
		||||
  webview_flutter_platform_interface:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter_platform_interface
 | 
			
		||||
      sha256: "6341f92977609be71391f4d4dcd64bfaa8ac657af1dfb2e231b5c1724e8c6c36"
 | 
			
		||||
      sha256: "78715dc442b7849dbde74e92bb67de1cecf5addf95531c5fb474e72f5fe9a507"
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "2.2.0"
 | 
			
		||||
    version: "2.3.0"
 | 
			
		||||
  webview_flutter_wkwebview:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    description:
 | 
			
		||||
      name: webview_flutter_wkwebview
 | 
			
		||||
      sha256: "2ef3f65fd49061c18e4d837a411308f2850417f2d0a7c11aad2c3857bee12c18"
 | 
			
		||||
      sha256: c94d242d8cbe1012c06ba7ac790c46d6e6b68723b7d34f8c74ed19f68d166f49
 | 
			
		||||
      url: "https://pub.dev"
 | 
			
		||||
    source: hosted
 | 
			
		||||
    version: "3.3.0"
 | 
			
		||||
    version: "3.4.0"
 | 
			
		||||
  win32:
 | 
			
		||||
    dependency: transitive
 | 
			
		||||
    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
 | 
			
		||||
# 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.
 | 
			
		||||
version: 0.11.31+153 # When changing this, update the tag in main() accordingly
 | 
			
		||||
version: 0.11.33+155 # When changing this, update the tag in main() accordingly
 | 
			
		||||
 | 
			
		||||
environment:
 | 
			
		||||
  sdk: '>=2.18.2 <3.0.0'
 | 
			
		||||
@@ -49,7 +49,7 @@ dependencies:
 | 
			
		||||
  permission_handler: ^10.0.0
 | 
			
		||||
  fluttertoast: ^8.0.9
 | 
			
		||||
  device_info_plus: ^8.0.0
 | 
			
		||||
  file_picker: ^5.1.0
 | 
			
		||||
  file_picker: ^5.2.10
 | 
			
		||||
  animations: ^2.0.4
 | 
			
		||||
  install_plugin_v2: ^1.0.0
 | 
			
		||||
  share_plus: ^6.0.1
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user