Add confirm=? query-param to delete-user route and confirm dialog in flutter [skip-tests]
This commit is contained in:
@@ -1,5 +1,7 @@
|
||||
import 'dart:async';
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:action_slider/action_slider.dart';
|
||||
import 'package:firebase_messaging/firebase_messaging.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
||||
@@ -527,17 +529,30 @@ class _AccountRootPageState extends State<AccountRootPage> {
|
||||
final acc = AppAuth();
|
||||
if (!acc.isAuth()) return;
|
||||
|
||||
try {
|
||||
TODO ASK BEFORE DELETING TEH FUCKING USER !!!!!!!
|
||||
final r1 = await UIDialogs.showConfirmDialog(context, "Delete Account?", text: "Are you sure you want to delete your account?", okText: "Delete", cancelText: "Cancel");
|
||||
if (!r1) return;
|
||||
|
||||
final r2 = await UIDialogs.showConfirmDialog(context, "Really sure?", text: "Are you really sure you want to delete your account?.\n\nThis includes all your messages and channels etc.\nThis action cannot be undone!", okText: "Delete", cancelText: "Cancel");
|
||||
if (!r2) return;
|
||||
|
||||
final r3 = await UIDialogs.showTextInput(context, "Please input 'Delete' to delete your Account.", "");
|
||||
if (r3 == null) return;
|
||||
if (r3.trim().toLowerCase() != 'delete') return;
|
||||
|
||||
final r4 = await this._showDeleteDialog(context);
|
||||
if (!r4) return;
|
||||
|
||||
final r5 = await this._showDeleteAccountWaitDialog(context);
|
||||
if (!r5) return;
|
||||
|
||||
try {
|
||||
await APIClient.deleteUser(acc, acc.userID!);
|
||||
|
||||
Toaster.info('Logout', 'Successfully logged out');
|
||||
|
||||
//TODO clear messages/channels/etc in open views
|
||||
acc.clear();
|
||||
await acc.save();
|
||||
|
||||
//TODO clear messages/channels/etc in open views
|
||||
} catch (exc, trace) {
|
||||
Toaster.error("Error", 'Failed to delete user');
|
||||
ApplicationLog.error('Failed to delete user: ' + exc.toString(), trace: trace);
|
||||
@@ -570,4 +585,136 @@ class _AccountRootPageState extends State<AccountRootPage> {
|
||||
ApplicationLog.error('Failed to update username: ' + exc.toString(), trace: trace);
|
||||
}
|
||||
}
|
||||
|
||||
Future<bool> _showDeleteDialog(BuildContext context) {
|
||||
return showDialog<bool>(
|
||||
context: context,
|
||||
builder: (context) => AlertDialog(
|
||||
title: Text("Delete Account?"),
|
||||
content: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
SizedBox(
|
||||
child: ActionSlider.standard(
|
||||
sliderBehavior: SliderBehavior.stretch,
|
||||
width: 300,
|
||||
backgroundColor: Colors.white,
|
||||
toggleColor: Colors.red,
|
||||
action: (controller) => Navigator.of(context).pop(true),
|
||||
child: const Text('Slide to delete'),
|
||||
),
|
||||
width: 300,
|
||||
height: 65,
|
||||
),
|
||||
],
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.of(context).pop(false),
|
||||
child: Text('Cancel'),
|
||||
),
|
||||
],
|
||||
),
|
||||
).then((value) => value ?? false);
|
||||
}
|
||||
|
||||
Future<bool> _showDeleteAccountWaitDialog(BuildContext context) {
|
||||
final completer = Completer<bool>();
|
||||
|
||||
bool isTimerActive = true;
|
||||
|
||||
const int totalSeconds = 20;
|
||||
int secondsRemaining = totalSeconds;
|
||||
double percentageRemaining = 1.0;
|
||||
|
||||
late Timer timer;
|
||||
|
||||
showDialog<void>(
|
||||
context: context,
|
||||
barrierDismissible: false,
|
||||
builder: (BuildContext dialogContext) {
|
||||
void Function(void Function())? setStateInner = null;
|
||||
|
||||
final t0 = DateTime.now();
|
||||
|
||||
timer = Timer.periodic(const Duration(milliseconds: 50), (t) {
|
||||
setStateInner?.call(() {
|
||||
percentageRemaining = 1 - (DateTime.now().millisecondsSinceEpoch - t0.millisecondsSinceEpoch) / (totalSeconds * 1000.0);
|
||||
secondsRemaining = (totalSeconds * percentageRemaining).ceil();
|
||||
|
||||
if (secondsRemaining <= 0) {
|
||||
t.cancel();
|
||||
isTimerActive = false;
|
||||
|
||||
// Close the dialog and return true for successful deletion
|
||||
Navigator.of(dialogContext).pop();
|
||||
if (!completer.isCompleted) {
|
||||
completer.complete(true);
|
||||
}
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
return StatefulBuilder(
|
||||
builder: (context, setState) {
|
||||
setStateInner = setState;
|
||||
return AlertDialog(
|
||||
title: const Text('Delete Account'),
|
||||
content: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Text(
|
||||
'Your account will be deleted in $secondsRemaining seconds.',
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
const SizedBox(height: 20),
|
||||
SizedBox(
|
||||
height: 100,
|
||||
width: 100,
|
||||
child: CircularProgressIndicator(
|
||||
value: 1 - percentageRemaining,
|
||||
strokeWidth: 8.0,
|
||||
valueColor: const AlwaysStoppedAnimation<Color>(Colors.red),
|
||||
backgroundColor: Colors.grey[300],
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
onPressed: () {
|
||||
// Cancel the timer
|
||||
if (timer.isActive) {
|
||||
timer.cancel();
|
||||
}
|
||||
isTimerActive = false;
|
||||
|
||||
// Close the dialog and return false for cancellation
|
||||
Navigator.of(dialogContext).pop();
|
||||
if (!completer.isCompleted) {
|
||||
completer.complete(false);
|
||||
}
|
||||
},
|
||||
child: const Text('CANCEL'),
|
||||
),
|
||||
],
|
||||
);
|
||||
},
|
||||
);
|
||||
},
|
||||
).then((_) {
|
||||
// Ensure timer is cancelled if dialog is dismissed
|
||||
if (timer.isActive) {
|
||||
timer.cancel();
|
||||
}
|
||||
|
||||
// If the completer hasn't been completed yet (e.g., if the dialog is dismissed),
|
||||
// complete it with false
|
||||
if (!completer.isCompleted && isTimerActive) {
|
||||
completer.complete(false);
|
||||
}
|
||||
});
|
||||
|
||||
return completer.future;
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user