1
0
mirror of https://github.com/go-vikunja/app synced 2024-06-02 18:49:47 +00:00
app-mirror-github/lib/global.dart

253 lines
8.0 KiB
Dart
Raw Permalink Normal View History

import 'dart:developer' as dev;
2024-04-05 12:35:23 +00:00
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
2022-07-15 14:25:16 +00:00
import 'package:vikunja_app/api/bucket_implementation.dart';
2018-09-22 20:56:16 +00:00
import 'package:vikunja_app/api/client.dart';
2021-06-04 09:34:25 +00:00
import 'package:vikunja_app/api/label_task.dart';
import 'package:vikunja_app/api/label_task_bulk.dart';
import 'package:vikunja_app/api/labels.dart';
import 'package:vikunja_app/api/server_implementation.dart';
2018-09-22 20:56:16 +00:00
import 'package:vikunja_app/api/task_implementation.dart';
import 'package:vikunja_app/api/user_implementation.dart';
import 'package:vikunja_app/api/version_check.dart';
import 'package:vikunja_app/managers/notifications.dart';
2018-09-22 20:56:16 +00:00
import 'package:vikunja_app/managers/user.dart';
import 'package:vikunja_app/models/user.dart';
import 'package:vikunja_app/service/services.dart';
import 'package:timezone/data/latest_all.dart' as tz;
2023-01-07 15:55:40 +00:00
import 'package:workmanager/workmanager.dart';
2022-04-22 19:39:01 +00:00
2023-07-22 20:54:50 +00:00
import 'api/project.dart';
2023-05-16 23:22:59 +00:00
import 'main.dart';
class VikunjaGlobal extends StatefulWidget {
final Widget child;
final Widget login;
2022-08-27 21:04:43 +00:00
VikunjaGlobal({required this.child, required this.login});
@override
VikunjaGlobalState createState() => VikunjaGlobalState();
static VikunjaGlobalState of(BuildContext context) {
var widget =
2023-05-16 23:22:59 +00:00
context.dependOnInheritedWidgetOfExactType<VikunjaGlobalInherited>();
2022-08-27 21:04:43 +00:00
return widget!.data;
}
}
class VikunjaGlobalState extends State<VikunjaGlobal> {
final FlutterSecureStorage _storage = new FlutterSecureStorage();
2022-08-27 21:04:43 +00:00
User? _currentUser;
bool _loading = true;
bool expired = false;
2022-08-27 21:04:43 +00:00
late Client _client;
UserService? _newUserService;
NotificationClass _notificationClass = NotificationClass();
2022-08-27 21:04:43 +00:00
User? get currentUser => _currentUser;
2019-03-18 16:56:15 +00:00
Client get client => _client;
2023-05-16 23:22:59 +00:00
GlobalKey<ScaffoldMessengerState> get snackbarKey => globalSnackbarKey;
UserManager get userManager => new UserManager(_storage);
2019-03-18 16:56:15 +00:00
2022-08-27 21:04:43 +00:00
UserService? get newUserService => _newUserService;
ServerService get serverService => new ServerAPIService(client);
2019-03-18 16:56:15 +00:00
SettingsManager get settingsManager => new SettingsManager(_storage);
VersionChecker get versionChecker => new VersionChecker(snackbarKey);
ProjectService get projectService => new ProjectAPIService(client, _storage);
2023-07-22 20:54:50 +00:00
TaskService get taskService => new TaskAPIService(client);
2019-03-18 16:56:15 +00:00
2022-07-15 14:25:16 +00:00
BucketService get bucketService => new BucketAPIService(client);
2022-04-15 18:14:04 +00:00
TaskServiceOptions get taskServiceOptions => new TaskServiceOptions();
NotificationClass get notifications => _notificationClass;
2021-06-04 09:34:25 +00:00
LabelService get labelService => new LabelAPIService(client);
LabelTaskService get labelTaskService => new LabelTaskAPIService(client);
LabelTaskBulkAPIService get labelTaskBulkService =>
new LabelTaskBulkAPIService(client);
2022-08-27 21:04:43 +00:00
late String currentTimeZone;
2023-01-07 15:55:40 +00:00
void updateWorkmanagerDuration() {
2024-04-05 12:35:23 +00:00
if (kIsWeb) {
return;
}
2023-01-07 15:55:40 +00:00
Workmanager().cancelAll().then((value) {
settingsManager.getWorkmanagerDuration().then((duration) {
if (duration.inMinutes > 0) {
Workmanager().registerPeriodicTask("update-tasks", "update-tasks",
frequency: duration,
constraints: Constraints(networkType: NetworkType.connected),
initialDelay: Duration(seconds: 15),
inputData: {
"client_token": client.token,
"client_base": client.base
});
2023-01-07 15:55:40 +00:00
}
Workmanager().registerPeriodicTask("refresh-token", "refresh-token",
frequency: Duration(hours: 12),
constraints: Constraints(networkType: NetworkType.connected),
initialDelay: Duration(seconds: 15));
2023-01-07 15:55:40 +00:00
});
});
}
@override
void initState() {
super.initState();
2023-05-16 23:22:59 +00:00
_client = Client(snackbarKey);
settingsManager
.getIgnoreCertificates()
.then((value) => client.reloadIgnoreCerts(value == "1"));
_newUserService = UserAPIService(client);
_loadCurrentUser();
tz.initializeTimeZones();
notifications.notificationInitializer();
settingsManager.getVersionNotifications().then((value) {
if (value == "1") {
versionChecker.postVersionCheckSnackbar();
}
});
}
2022-08-27 21:04:43 +00:00
void changeUser(User newUser, {String? token, String? base}) async {
setState(() {
_loading = true;
});
if (token == null) {
token = await _storage.read(key: newUser.id.toString());
} else {
// Write new token to secure storage
await _storage.write(key: newUser.id.toString(), value: token);
}
2018-09-17 13:35:57 +00:00
if (base == null) {
base = await _storage.read(key: "${newUser.id.toString()}_base");
} else {
// Write new base to secure storage
await _storage.write(key: "${newUser.id.toString()}_base", value: base);
}
// Set current user in storage
await _storage.write(key: 'currentUser', value: newUser.id.toString());
client.configure(token: token, base: base, authenticated: true);
updateWorkmanagerDuration();
setState(() {
_currentUser = newUser;
_loading = false;
});
}
void logoutUser(BuildContext context) async {
// _storage.deleteAll().then((_) {
var userId = await _storage.read(key: "currentUser");
await _storage.delete(key: userId!); //delete token
await _storage.delete(key: "${userId}_base");
setState(() {
client.reset();
_currentUser = null;
});
/* }).catchError((err) {
2022-04-10 13:31:56 +00:00
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
2023-05-16 19:08:58 +00:00
content: Text('An error occurred while logging out!'),
2019-03-18 16:56:15 +00:00
));
});*/
2019-03-18 16:56:15 +00:00
}
void _loadCurrentUser() async {
var currentUser = await _storage.read(key: 'currentUser');
2018-09-17 13:35:57 +00:00
if (currentUser == null) {
setState(() {
_loading = false;
});
return;
}
var token = await _storage.read(key: currentUser);
var base = await _storage.read(key: '${currentUser}_base');
if (token == null || base == null) {
setState(() {
_loading = false;
});
return;
}
2022-05-07 15:44:11 +00:00
client.configure(token: token, base: base, authenticated: true);
2022-09-03 15:43:16 +00:00
User loadedCurrentUser;
try {
loadedCurrentUser = await UserAPIService(client).getCurrentUser();
// load new token from server to avoid expiration
String? newToken = await newUserService?.getToken();
if (newToken != null) {
_storage.write(key: currentUser, value: newToken);
client.configure(token: newToken);
}
} on ApiException catch (e) {
dev.log("Error code: " + e.errorCode.toString(), level: 1000);
if (e.errorCode ~/ 100 == 4) {
client.authenticated = false;
if (e.errorCode == 401) {
// token has expired, but we can reuse username and base. user just has to enter password again
expired = true;
}
setState(() {
client.authenticated = false;
_currentUser = null;
_loading = false;
});
return;
}
2022-09-03 15:43:16 +00:00
loadedCurrentUser = User(id: int.parse(currentUser), username: '');
} catch (otherExceptions) {
2022-09-03 15:43:16 +00:00
loadedCurrentUser = User(id: int.parse(currentUser), username: '');
}
updateWorkmanagerDuration();
2018-09-17 13:35:57 +00:00
setState(() {
_currentUser = loadedCurrentUser;
_loading = false;
});
}
@override
Widget build(BuildContext context) {
if (_loading) {
return new Center(child: new CircularProgressIndicator());
}
if (client.authenticated) {
notifications.scheduleDueNotifications(taskService);
}
2023-05-16 23:22:59 +00:00
return new VikunjaGlobalInherited(
data: this,
2022-08-27 21:04:43 +00:00
key: UniqueKey(),
child: !client.authenticated ? widget.login : widget.child,
);
}
}
2023-05-16 23:22:59 +00:00
class VikunjaGlobalInherited extends InheritedWidget {
final VikunjaGlobalState data;
2023-05-16 23:22:59 +00:00
VikunjaGlobalInherited({Key? key, required this.data, required Widget child})
: super(key: key, child: child);
@override
2023-05-16 23:22:59 +00:00
bool updateShouldNotify(VikunjaGlobalInherited oldWidget) {
2018-09-16 19:47:53 +00:00
return (data.currentUser != null &&
2022-08-27 21:04:43 +00:00
data.currentUser!.id != oldWidget.data.currentUser!.id) ||
data.client != oldWidget.data.client;
}
}