app_update_dialog.dart 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. import 'package:flutter/material.dart';
  2. import 'package:logging/logging.dart';
  3. // import 'package:open_file/open_file.dart';
  4. import 'package:photos/core/configuration.dart';
  5. import 'package:photos/core/network/network.dart';
  6. import 'package:photos/ente_theme_data.dart';
  7. import "package:photos/generated/l10n.dart";
  8. import 'package:photos/services/update_service.dart';
  9. import 'package:photos/theme/ente_theme.dart';
  10. import 'package:url_launcher/url_launcher_string.dart';
  11. class AppUpdateDialog extends StatefulWidget {
  12. final LatestVersionInfo? latestVersionInfo;
  13. const AppUpdateDialog(this.latestVersionInfo, {Key? key}) : super(key: key);
  14. @override
  15. State<AppUpdateDialog> createState() => _AppUpdateDialogState();
  16. }
  17. class _AppUpdateDialogState extends State<AppUpdateDialog> {
  18. @override
  19. Widget build(BuildContext context) {
  20. final List<Widget> changelog = [];
  21. final enteTextTheme = getEnteTextTheme(context);
  22. final enteColor = getEnteColorScheme(context);
  23. for (final log in widget.latestVersionInfo!.changelog) {
  24. changelog.add(
  25. Padding(
  26. padding: const EdgeInsets.fromLTRB(0, 4, 0, 4),
  27. child: Row(
  28. mainAxisAlignment: MainAxisAlignment.start,
  29. crossAxisAlignment: CrossAxisAlignment.start,
  30. children: [
  31. Text(
  32. "- ",
  33. style: enteTextTheme.small.copyWith(color: enteColor.textMuted),
  34. ),
  35. Flexible(
  36. child: Text(
  37. log,
  38. softWrap: true,
  39. style:
  40. enteTextTheme.small.copyWith(color: enteColor.textMuted),
  41. ),
  42. )
  43. ],
  44. ),
  45. ),
  46. );
  47. }
  48. final content = Column(
  49. crossAxisAlignment: CrossAxisAlignment.start,
  50. mainAxisSize: MainAxisSize.min,
  51. children: [
  52. Text(
  53. S.of(context).aNewVersionOfEnteIsAvailable,
  54. style: enteTextTheme.body.copyWith(color: enteColor.textMuted),
  55. ),
  56. const Padding(padding: EdgeInsets.all(8)),
  57. Column(
  58. crossAxisAlignment: CrossAxisAlignment.start,
  59. children: changelog,
  60. ),
  61. const Padding(padding: EdgeInsets.all(8)),
  62. SizedBox(
  63. width: double.infinity,
  64. height: 56,
  65. child: OutlinedButton(
  66. style: Theme.of(context).outlinedButtonTheme.style!.copyWith(
  67. textStyle: MaterialStateProperty.resolveWith<TextStyle>(
  68. (Set<MaterialState> states) {
  69. return enteTextTheme.bodyBold;
  70. },
  71. ),
  72. ),
  73. onPressed: () async {
  74. Navigator.pop(context);
  75. showDialog(
  76. context: context,
  77. builder: (BuildContext context) {
  78. return ApkDownloaderDialog(widget.latestVersionInfo);
  79. },
  80. barrierDismissible: false,
  81. );
  82. },
  83. child: Text(
  84. S.of(context).update,
  85. ),
  86. ),
  87. ),
  88. const Padding(padding: EdgeInsets.all(8)),
  89. Center(
  90. child: InkWell(
  91. child: Text(
  92. S.of(context).installManually,
  93. style: Theme.of(context)
  94. .textTheme
  95. .caption!
  96. .copyWith(decoration: TextDecoration.underline),
  97. ),
  98. onTap: () => launchUrlString(
  99. widget.latestVersionInfo!.url,
  100. mode: LaunchMode.externalApplication,
  101. ),
  102. ),
  103. )
  104. ],
  105. );
  106. final shouldForceUpdate =
  107. UpdateService.instance.shouldForceUpdate(widget.latestVersionInfo!);
  108. return WillPopScope(
  109. onWillPop: () async => !shouldForceUpdate,
  110. child: AlertDialog(
  111. key: const ValueKey("updateAppDialog"),
  112. title: Column(
  113. crossAxisAlignment: CrossAxisAlignment.start,
  114. children: [
  115. Icon(
  116. Icons.auto_awesome_outlined,
  117. size: 48,
  118. color: enteColor.strokeMuted,
  119. ),
  120. const SizedBox(
  121. height: 16,
  122. ),
  123. Text(
  124. shouldForceUpdate
  125. ? S.of(context).criticalUpdateAvailable
  126. : S.of(context).updateAvailable,
  127. style: enteTextTheme.h3Bold,
  128. ),
  129. ],
  130. ),
  131. content: content,
  132. ),
  133. );
  134. }
  135. }
  136. class ApkDownloaderDialog extends StatefulWidget {
  137. final LatestVersionInfo? versionInfo;
  138. const ApkDownloaderDialog(this.versionInfo, {Key? key}) : super(key: key);
  139. @override
  140. State<ApkDownloaderDialog> createState() => _ApkDownloaderDialogState();
  141. }
  142. class _ApkDownloaderDialogState extends State<ApkDownloaderDialog> {
  143. String? _saveUrl;
  144. double? _downloadProgress;
  145. @override
  146. void initState() {
  147. super.initState();
  148. _saveUrl = Configuration.instance.getTempDirectory() +
  149. "ente-" +
  150. widget.versionInfo!.name +
  151. ".apk";
  152. _downloadApk();
  153. }
  154. @override
  155. Widget build(BuildContext context) {
  156. return WillPopScope(
  157. onWillPop: () async => false,
  158. child: AlertDialog(
  159. title: Text(
  160. S.of(context).downloading,
  161. style: const TextStyle(
  162. fontSize: 16,
  163. ),
  164. textAlign: TextAlign.center,
  165. ),
  166. content: LinearProgressIndicator(
  167. value: _downloadProgress,
  168. valueColor: AlwaysStoppedAnimation<Color>(
  169. Theme.of(context).colorScheme.greenAlternative,
  170. ),
  171. ),
  172. ),
  173. );
  174. }
  175. Future<void> _downloadApk() async {
  176. try {
  177. await NetworkClient.instance.getDio().download(
  178. widget.versionInfo!.url,
  179. _saveUrl,
  180. onReceiveProgress: (count, _) {
  181. setState(() {
  182. _downloadProgress = count / widget.versionInfo!.size;
  183. });
  184. },
  185. );
  186. Navigator.of(context, rootNavigator: true).pop('dialog');
  187. // OpenFile.open(_saveUrl);
  188. } catch (e) {
  189. Logger("ApkDownloader").severe(e);
  190. final AlertDialog alert = AlertDialog(
  191. title: Text(S.of(context).sorry),
  192. content: Text(S.of(context).theDownloadCouldNotBeCompleted),
  193. actions: [
  194. TextButton(
  195. child: const Text(
  196. "Ignore",
  197. style: TextStyle(
  198. color: Colors.white,
  199. ),
  200. ),
  201. onPressed: () {
  202. Navigator.of(context, rootNavigator: true).pop('dialog');
  203. Navigator.of(context, rootNavigator: true).pop('dialog');
  204. },
  205. ),
  206. TextButton(
  207. child: Text(
  208. S.of(context).retry,
  209. style: TextStyle(
  210. color: Theme.of(context).colorScheme.greenAlternative,
  211. ),
  212. ),
  213. onPressed: () {
  214. Navigator.of(context, rootNavigator: true).pop('dialog');
  215. Navigator.of(context, rootNavigator: true).pop('dialog');
  216. showDialog(
  217. context: context,
  218. builder: (BuildContext context) {
  219. return ApkDownloaderDialog(widget.versionInfo);
  220. },
  221. barrierDismissible: false,
  222. );
  223. },
  224. ),
  225. ],
  226. );
  227. showDialog(
  228. context: context,
  229. builder: (BuildContext context) {
  230. return alert;
  231. },
  232. barrierColor: Colors.black87,
  233. );
  234. return;
  235. }
  236. }
  237. }