2020-05-02 08:51:34 +02:00
|
|
|
import 'dart:typed_data';
|
|
|
|
|
2020-05-01 15:12:55 +02:00
|
|
|
import 'package:comunic/helpers/groups_helper.dart';
|
|
|
|
import 'package:comunic/models/advanced_group_info.dart';
|
2020-05-01 20:52:24 +02:00
|
|
|
import 'package:comunic/models/group.dart';
|
2020-05-02 09:45:03 +02:00
|
|
|
import 'package:comunic/ui/dialogs/input_user_password_dialog.dart';
|
2020-05-01 20:52:24 +02:00
|
|
|
import 'package:comunic/ui/dialogs/multi_choices_dialog.dart';
|
2020-05-01 20:10:25 +02:00
|
|
|
import 'package:comunic/ui/dialogs/virtual_directory_dialog.dart';
|
2020-05-05 13:21:37 +02:00
|
|
|
import 'package:comunic/ui/routes/main_route/main_route.dart';
|
2020-05-01 15:12:55 +02:00
|
|
|
import 'package:comunic/ui/widgets/async_screen_widget.dart';
|
|
|
|
import 'package:comunic/ui/widgets/comunic_back_button_widget.dart';
|
2020-05-02 08:51:34 +02:00
|
|
|
import 'package:comunic/ui/widgets/group_icon_widget.dart';
|
2020-05-01 15:12:55 +02:00
|
|
|
import 'package:comunic/ui/widgets/safe_state.dart';
|
2020-05-01 20:52:24 +02:00
|
|
|
import 'package:comunic/ui/widgets/settings/multi_choices_settings_tile.dart';
|
2020-05-01 15:39:54 +02:00
|
|
|
import 'package:comunic/ui/widgets/settings/text_settings_edit_tile.dart';
|
2020-05-02 08:51:34 +02:00
|
|
|
import 'package:comunic/utils/files_utils.dart';
|
2020-05-01 20:19:22 +02:00
|
|
|
import 'package:comunic/utils/input_utils.dart';
|
2020-05-01 15:12:55 +02:00
|
|
|
import 'package:comunic/utils/intl_utils.dart';
|
2020-05-01 15:39:54 +02:00
|
|
|
import 'package:comunic/utils/ui_utils.dart';
|
2020-05-01 15:12:55 +02:00
|
|
|
import 'package:flutter/material.dart';
|
2020-05-02 09:05:38 +02:00
|
|
|
import 'package:identicon/identicon.dart';
|
|
|
|
import 'package:random_string/random_string.dart';
|
2020-05-01 15:12:55 +02:00
|
|
|
import 'package:settings_ui/settings_ui.dart';
|
|
|
|
|
|
|
|
/// Groups settings screen
|
|
|
|
///
|
|
|
|
/// @author Pierre Hubert
|
|
|
|
|
|
|
|
class GroupSettingsScreen extends StatefulWidget {
|
|
|
|
final int groupID;
|
|
|
|
|
|
|
|
const GroupSettingsScreen({Key key, @required this.groupID})
|
|
|
|
: assert(groupID != null),
|
|
|
|
super(key: key);
|
|
|
|
|
|
|
|
@override
|
|
|
|
_GroupSettingsScreenState createState() => _GroupSettingsScreenState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _GroupSettingsScreenState extends SafeState<GroupSettingsScreen> {
|
|
|
|
AdvancedGroupInfo _groupSettings;
|
|
|
|
|
2020-05-01 15:39:54 +02:00
|
|
|
final _key = GlobalKey<AsyncScreenWidgetState>();
|
|
|
|
|
2020-05-01 15:12:55 +02:00
|
|
|
Future<void> _refresh() async {
|
|
|
|
_groupSettings = await GroupsHelper().getSettings(widget.groupID);
|
|
|
|
}
|
|
|
|
|
2020-05-01 15:39:54 +02:00
|
|
|
Future<void> _updateSettings() async {
|
|
|
|
try {
|
|
|
|
await GroupsHelper.setSettings(_groupSettings);
|
|
|
|
} catch (e, stack) {
|
|
|
|
print("Could not update group settings! $e\n$stack");
|
|
|
|
showSimpleSnack(context, tr("Could not update group settings!"));
|
|
|
|
}
|
|
|
|
|
|
|
|
_key.currentState.refresh();
|
|
|
|
}
|
|
|
|
|
2020-05-01 15:12:55 +02:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Scaffold(
|
|
|
|
appBar: AppBar(
|
|
|
|
leading: ComunicBackButton(),
|
|
|
|
title: Text(tr("Group settings")),
|
|
|
|
),
|
|
|
|
body: _buildBody(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildBody() {
|
|
|
|
return AsyncScreenWidget(
|
2020-05-01 15:39:54 +02:00
|
|
|
key: _key,
|
2020-05-01 15:12:55 +02:00
|
|
|
onReload: _refresh,
|
|
|
|
onBuild: _buildContent,
|
|
|
|
errorMessage: tr("Could not get group settings!"),
|
2020-05-01 15:39:54 +02:00
|
|
|
showOldDataWhileUpdating: true,
|
2020-05-01 15:12:55 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildContent() {
|
|
|
|
return SettingsList(
|
2020-05-01 20:52:24 +02:00
|
|
|
sections: [
|
|
|
|
_buildGeneralSection(),
|
|
|
|
_buildAccessRestrictions(),
|
2020-05-02 09:45:03 +02:00
|
|
|
_buildGroupLogoArea(),
|
|
|
|
_buildDangerZone(),
|
2020-05-01 20:52:24 +02:00
|
|
|
],
|
2020-05-01 15:12:55 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildGeneralSection() {
|
|
|
|
return SettingsSection(
|
|
|
|
title: tr("General information"),
|
|
|
|
tiles: [
|
2020-05-01 15:39:54 +02:00
|
|
|
// Group ID
|
2020-05-01 15:12:55 +02:00
|
|
|
SettingsTile(
|
|
|
|
title: tr("Group ID"),
|
|
|
|
subtitle: _groupSettings.id.toString(),
|
|
|
|
),
|
2020-05-01 15:39:54 +02:00
|
|
|
|
|
|
|
// Group name
|
|
|
|
TextEditSettingsTile(
|
|
|
|
title: tr("Group name"),
|
|
|
|
currValue: _groupSettings.name,
|
|
|
|
onChanged: (s) {
|
|
|
|
_groupSettings.name = s;
|
|
|
|
_updateSettings();
|
|
|
|
}),
|
2020-05-01 20:10:25 +02:00
|
|
|
|
|
|
|
// Group virtual directory
|
|
|
|
SettingsTile(
|
|
|
|
title: tr("Virtual directory (optional)"),
|
|
|
|
subtitle: _groupSettings.virtualDirectory,
|
|
|
|
onTap: () async {
|
|
|
|
final newDir = await showVirtualDirectoryDialog(
|
|
|
|
context: context,
|
|
|
|
initialDirectory: _groupSettings.virtualDirectory,
|
|
|
|
id: _groupSettings.id,
|
|
|
|
type: VirtualDirectoryTargetType.GROUP,
|
|
|
|
);
|
|
|
|
|
|
|
|
if (newDir == null) return;
|
|
|
|
|
|
|
|
_groupSettings.virtualDirectory = newDir;
|
|
|
|
_updateSettings();
|
|
|
|
},
|
|
|
|
),
|
2020-05-01 20:19:22 +02:00
|
|
|
|
|
|
|
// Group URL
|
|
|
|
TextEditSettingsTile(
|
|
|
|
title: tr("Group URL (optional)"),
|
|
|
|
currValue: _groupSettings.url,
|
|
|
|
checkInput: validateUrl,
|
|
|
|
allowEmptyValues: true,
|
|
|
|
onChanged: (s) {
|
|
|
|
_groupSettings.url = s;
|
|
|
|
_updateSettings();
|
|
|
|
},
|
2020-05-01 20:30:26 +02:00
|
|
|
),
|
|
|
|
|
|
|
|
// Group description
|
|
|
|
TextEditSettingsTile(
|
|
|
|
title: tr("Group description (optional)"),
|
|
|
|
currValue: _groupSettings.description,
|
|
|
|
maxLines: 3,
|
|
|
|
maxLength: 255,
|
|
|
|
allowEmptyValues: true,
|
|
|
|
onChanged: (s) {
|
|
|
|
_groupSettings.description = s;
|
|
|
|
_updateSettings();
|
|
|
|
}),
|
2020-05-01 15:12:55 +02:00
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
2020-05-01 20:52:24 +02:00
|
|
|
|
|
|
|
List<MultiChoiceEntry<GroupVisibilityLevel>> get _visibilityLevels => [
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupVisibilityLevel.OPEN,
|
|
|
|
title: tr("Open group"),
|
|
|
|
subtitle:
|
|
|
|
tr("Group information & public posts are available to everyone."),
|
|
|
|
),
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupVisibilityLevel.PRIVATE,
|
2020-05-01 21:04:50 +02:00
|
|
|
title: tr("Private group"),
|
2020-05-01 20:52:24 +02:00
|
|
|
subtitle: tr("The group is accessible to accepted members only."),
|
|
|
|
),
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupVisibilityLevel.SECRETE,
|
2020-05-01 21:04:50 +02:00
|
|
|
title: tr("Secrete group"),
|
2020-05-01 20:52:24 +02:00
|
|
|
subtitle: tr("The group is visible only to invited members."),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
2020-05-01 21:04:50 +02:00
|
|
|
List<MultiChoiceEntry<GroupRegistrationLevel>> get _registrationLevels => [
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupRegistrationLevel.OPEN,
|
|
|
|
title: tr("Open registration"),
|
|
|
|
subtitle: tr(
|
|
|
|
"Everyone can choose to join the group without moderator approval"),
|
|
|
|
),
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupRegistrationLevel.MODERATED,
|
|
|
|
title: tr("Moderated registration"),
|
|
|
|
subtitle: tr(
|
|
|
|
"Everyone can request a membership, but a moderator review the request"),
|
|
|
|
),
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupRegistrationLevel.CLOSED,
|
|
|
|
title: tr("Closed registration"),
|
|
|
|
subtitle: tr(
|
|
|
|
"The only way to join the group is to be invited by a moderator"),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
2020-05-01 21:13:31 +02:00
|
|
|
List<MultiChoiceEntry<GroupPostCreationLevel>> get _postsCreationLevels => [
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupPostCreationLevel.MEMBERS,
|
|
|
|
title: tr("All members"),
|
|
|
|
subtitle:
|
|
|
|
tr("All the members of the group can create posts on the group"),
|
|
|
|
),
|
|
|
|
MultiChoiceEntry(
|
|
|
|
id: GroupPostCreationLevel.MODERATORS,
|
|
|
|
title: tr("Moderators only"),
|
|
|
|
subtitle: tr(
|
|
|
|
"Only moderators and administrators of the group can create posts on it"),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
2020-05-01 20:52:24 +02:00
|
|
|
Widget _buildAccessRestrictions() => SettingsSection(
|
|
|
|
title: tr("Access restrictions"),
|
|
|
|
tiles: [
|
2020-05-01 21:13:31 +02:00
|
|
|
// Group visibility
|
2020-05-01 20:52:24 +02:00
|
|
|
MultiChoicesSettingsTile(
|
|
|
|
title: tr("Group visibility"),
|
|
|
|
choices: _visibilityLevels,
|
|
|
|
currentValue: _groupSettings.visibilityLevel,
|
|
|
|
onChanged: (v) {
|
|
|
|
_groupSettings.visibilityLevel = v;
|
|
|
|
_updateSettings();
|
2020-05-01 21:04:50 +02:00
|
|
|
}),
|
2020-05-01 21:13:31 +02:00
|
|
|
|
|
|
|
// Group registration level
|
2020-05-01 21:04:50 +02:00
|
|
|
MultiChoicesSettingsTile(
|
|
|
|
title: tr("Group registration level"),
|
|
|
|
choices: _registrationLevels,
|
|
|
|
currentValue: _groupSettings.registrationLevel,
|
|
|
|
onChanged: (v) {
|
|
|
|
_groupSettings.registrationLevel = v;
|
|
|
|
_updateSettings();
|
|
|
|
}),
|
2020-05-01 21:13:31 +02:00
|
|
|
|
|
|
|
// Group posts creation levels
|
|
|
|
MultiChoicesSettingsTile(
|
|
|
|
title: tr("Posts creation level"),
|
|
|
|
choices: _postsCreationLevels,
|
|
|
|
currentValue: _groupSettings.postCreationLevel,
|
|
|
|
onChanged: (s) {
|
|
|
|
_groupSettings.postCreationLevel = s;
|
|
|
|
_updateSettings();
|
|
|
|
}),
|
2020-05-01 20:52:24 +02:00
|
|
|
],
|
|
|
|
);
|
2020-05-02 08:51:34 +02:00
|
|
|
|
|
|
|
Widget _buildGroupLogoArea() {
|
|
|
|
return SettingsSection(
|
|
|
|
title: tr("Group logo"),
|
|
|
|
tiles: [
|
|
|
|
// Current logo
|
|
|
|
SettingsTile(
|
|
|
|
title: tr("Current logo"),
|
|
|
|
leading: GroupIcon(group: _groupSettings),
|
|
|
|
),
|
|
|
|
|
|
|
|
// Upload a new logo
|
|
|
|
SettingsTile(
|
|
|
|
title: tr("Upload a new logo"),
|
|
|
|
onTap: _uploadNewLogo,
|
|
|
|
),
|
2020-05-02 09:05:38 +02:00
|
|
|
|
|
|
|
// Generate a new random logo
|
|
|
|
SettingsTile(
|
|
|
|
title: tr("Generate a new random logo"),
|
|
|
|
onTap: _generateRandomLogo,
|
2020-05-02 09:15:02 +02:00
|
|
|
),
|
|
|
|
|
|
|
|
// Delete current logo
|
|
|
|
SettingsTile(
|
|
|
|
title: tr("Delete logo"),
|
|
|
|
onTap: _deleteLogo,
|
|
|
|
),
|
2020-05-02 08:51:34 +02:00
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-05-02 09:05:38 +02:00
|
|
|
/// Upload a new logo for the group
|
2020-05-02 08:51:34 +02:00
|
|
|
void _uploadNewLogo() async {
|
|
|
|
try {
|
|
|
|
final logo = await pickImage(context);
|
|
|
|
final bytes = logo.readAsBytesSync();
|
|
|
|
await _doUploadLogo(bytes);
|
|
|
|
} catch (e, stack) {
|
|
|
|
print("Could not upload new logo! $e\n$stack");
|
|
|
|
showSimpleSnack(context, tr("Could not upload new logo!"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-02 09:05:38 +02:00
|
|
|
/// Generate a new random logo for the group
|
|
|
|
void _generateRandomLogo() async {
|
|
|
|
try {
|
|
|
|
final newLogo =
|
|
|
|
Identicon(rows: 10, cols: 10).generate(randomString(20), size: 100);
|
|
|
|
await _doUploadLogo(newLogo);
|
|
|
|
} catch (e, stack) {
|
|
|
|
print("Could not generate new logo! $e\n$stack");
|
|
|
|
showSimpleSnack(context, tr("Could not generate new random logo!"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-02 08:51:34 +02:00
|
|
|
Future<void> _doUploadLogo(Uint8List bytes) async {
|
|
|
|
await GroupsHelper.uploadNewLogo(_groupSettings.id, bytes);
|
|
|
|
_key.currentState.refresh();
|
|
|
|
}
|
2020-05-02 09:15:02 +02:00
|
|
|
|
|
|
|
/// Delete previous group logo
|
|
|
|
void _deleteLogo() async {
|
|
|
|
try {
|
|
|
|
if (!await showConfirmDialog(
|
|
|
|
context: context,
|
|
|
|
message: tr("Do you really want to delete the logo of this group ?")))
|
|
|
|
return;
|
|
|
|
|
|
|
|
await GroupsHelper.deleteLogo(_groupSettings.id);
|
|
|
|
_key.currentState.refresh();
|
|
|
|
} catch (e, s) {
|
|
|
|
print("Could not delete group logo! $e\n$s");
|
|
|
|
showSimpleSnack(context, tr("Could not delete group logo!"));
|
|
|
|
}
|
|
|
|
}
|
2020-05-02 09:45:03 +02:00
|
|
|
|
|
|
|
Widget _buildDangerZone() {
|
|
|
|
return SettingsSection(
|
|
|
|
title: tr("Danger zone"),
|
|
|
|
tiles: [
|
|
|
|
SettingsTile(
|
|
|
|
title: tr("Delete group"),
|
|
|
|
onTap: _deleteGroup,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Delete the group
|
|
|
|
void _deleteGroup() async {
|
|
|
|
try {
|
|
|
|
final password = await showUserPasswordDialog(context);
|
|
|
|
|
|
|
|
if (password == null) return;
|
|
|
|
|
|
|
|
if (!await showConfirmDialog(
|
|
|
|
context: context,
|
|
|
|
message: tr(
|
|
|
|
"Do you really want to delete this group ? All the posts related to it will be permanently deleted!")))
|
|
|
|
return;
|
|
|
|
|
|
|
|
await GroupsHelper.deleteGroup(_groupSettings.id, password);
|
|
|
|
|
|
|
|
MainController.of(context).popPage();
|
|
|
|
} catch (e, s) {
|
|
|
|
print("Could not delete the group! $e\n$s");
|
|
|
|
showSimpleSnack(context, tr("Could not delete the group"));
|
|
|
|
}
|
|
|
|
}
|
2020-05-01 15:12:55 +02:00
|
|
|
}
|