✨ Basic contact methods
This commit is contained in:
parent
172d0d24fb
commit
58421e5d5e
@ -5,14 +5,14 @@ meta {
|
|||||||
}
|
}
|
||||||
|
|
||||||
put {
|
put {
|
||||||
url: {{endpoint}}/cgi/id/reports/abuse/3/status
|
url: {{endpoint}}/cgi/id/reports/abuse/6/status
|
||||||
body: json
|
body: json
|
||||||
auth: inherit
|
auth: inherit
|
||||||
}
|
}
|
||||||
|
|
||||||
body:json {
|
body:json {
|
||||||
{
|
{
|
||||||
"status": "processed",
|
"status": "rejected",
|
||||||
"message": "相关附件已经进行评级处理,未来会将该项权限下放到帖主以及社区成员。"
|
"message": "Not a good reason"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -822,5 +822,18 @@
|
|||||||
"accountUnconfirmedResendSuccessful": "Email has been resent, you can resend it again in 60 minutes.",
|
"accountUnconfirmedResendSuccessful": "Email has been resent, you can resend it again in 60 minutes.",
|
||||||
"stickerPickerEmpty": "Sticker list is empty",
|
"stickerPickerEmpty": "Sticker list is empty",
|
||||||
"stickerPickerEmptyHint": "To start using stickers, you need to add a sticker pack first.",
|
"stickerPickerEmptyHint": "To start using stickers, you need to add a sticker pack first.",
|
||||||
"goto": "Go to {}"
|
"goto": "Go to {}",
|
||||||
|
"accountContactMethods": "Contact Methods",
|
||||||
|
"accountContactMethodsDescription": "Manage your contact methods.",
|
||||||
|
"accountContactMethodsNameEmail": "Email address",
|
||||||
|
"accountContactMethodsNamePhone": "Phone number",
|
||||||
|
"accountContactMethodsNameAddress": "Address",
|
||||||
|
"accountContactMethodsPrimary": "Primary",
|
||||||
|
"accountContactMethodsVerified": "Verified",
|
||||||
|
"accountContactMethodsPublic": "Public",
|
||||||
|
"accountContactMethodsAdd": "Add Contact Method",
|
||||||
|
"accountContactMethodsEdit": "Edit Contact Method",
|
||||||
|
"accountContactMethodsAddDescription": "Add a new contact method.",
|
||||||
|
"fieldContactContent": "Contact method",
|
||||||
|
"accountContactMethodsPublicHint": "This contact method will be displayed publicly on your profile."
|
||||||
}
|
}
|
||||||
|
@ -822,5 +822,18 @@
|
|||||||
"accountUnconfirmedResendSuccessful": "邮件已重新发送,你可以在 60 分钟后再重发一封。",
|
"accountUnconfirmedResendSuccessful": "邮件已重新发送,你可以在 60 分钟后再重发一封。",
|
||||||
"stickerPickerEmpty": "贴图列表为空",
|
"stickerPickerEmpty": "贴图列表为空",
|
||||||
"stickerPickerEmptyHint": "想要开始使用贴图,请先添加贴图包。",
|
"stickerPickerEmptyHint": "想要开始使用贴图,请先添加贴图包。",
|
||||||
"goto": "跳转到 {}"
|
"goto": "跳转到 {}",
|
||||||
|
"accountContactMethods": "联系方式",
|
||||||
|
"accountContactMethodsDescription": "管理你的联系方式。",
|
||||||
|
"accountContactMethodsNameEmail": "电子邮箱",
|
||||||
|
"accountContactMethodsNamePhone": "电话",
|
||||||
|
"accountContactMethodsNameAddress": "地址",
|
||||||
|
"accountContactMethodsPrimary": "主要的",
|
||||||
|
"accountContactMethodsVerified": "已验证",
|
||||||
|
"accountContactMethodsPublic": "公开的",
|
||||||
|
"accountContactMethodsAdd": "添加联系方式",
|
||||||
|
"accountContactMethodsEdit": "编辑联系方式",
|
||||||
|
"accountContactMethodsAddDescription": "添加新的联系方式。",
|
||||||
|
"fieldContactContent": "联系方式",
|
||||||
|
"accountContactMethodsPublicHint": "这个联系方式公开地显示在个人资料中。"
|
||||||
}
|
}
|
||||||
|
@ -822,5 +822,18 @@
|
|||||||
"accountUnconfirmedResendSuccessful": "郵件已重新發送,你可以在 60 分鐘後再重發一封。",
|
"accountUnconfirmedResendSuccessful": "郵件已重新發送,你可以在 60 分鐘後再重發一封。",
|
||||||
"stickerPickerEmpty": "貼圖列表為空",
|
"stickerPickerEmpty": "貼圖列表為空",
|
||||||
"stickerPickerEmptyHint": "想要開始使用貼圖,請先添加貼圖包。",
|
"stickerPickerEmptyHint": "想要開始使用貼圖,請先添加貼圖包。",
|
||||||
"goto": "跳轉到 {}"
|
"goto": "跳轉到 {}",
|
||||||
|
"accountContactMethods": "聯繫方式",
|
||||||
|
"accountContactMethodsDescription": "管理你的聯繫方式。",
|
||||||
|
"accountContactMethodsNameEmail": "電子郵箱",
|
||||||
|
"accountContactMethodsNamePhone": "電話",
|
||||||
|
"accountContactMethodsNameAddress": "地址",
|
||||||
|
"accountContactMethodsPrimary": "主要的",
|
||||||
|
"accountContactMethodsVerified": "已驗證",
|
||||||
|
"accountContactMethodsPublic": "公開的",
|
||||||
|
"accountContactMethodsAdd": "添加聯繫方式",
|
||||||
|
"accountContactMethodsEdit": "編輯聯繫方式",
|
||||||
|
"accountContactMethodsAddDescription": "添加新的聯繫方式。",
|
||||||
|
"fieldContactContent": "聯繫方式",
|
||||||
|
"accountContactMethodsPublicHint": "這個聯繫方式公開地顯示在個人資料中。"
|
||||||
}
|
}
|
||||||
|
@ -822,5 +822,18 @@
|
|||||||
"accountUnconfirmedResendSuccessful": "郵件已重新發送,你可以在 60 分鐘後再重發一封。",
|
"accountUnconfirmedResendSuccessful": "郵件已重新發送,你可以在 60 分鐘後再重發一封。",
|
||||||
"stickerPickerEmpty": "貼圖列表為空",
|
"stickerPickerEmpty": "貼圖列表為空",
|
||||||
"stickerPickerEmptyHint": "想要開始使用貼圖,請先添加貼圖包。",
|
"stickerPickerEmptyHint": "想要開始使用貼圖,請先添加貼圖包。",
|
||||||
"goto": "跳轉到 {}"
|
"goto": "跳轉到 {}",
|
||||||
|
"accountContactMethods": "聯繫方式",
|
||||||
|
"accountContactMethodsDescription": "管理你的聯繫方式。",
|
||||||
|
"accountContactMethodsNameEmail": "電子郵箱",
|
||||||
|
"accountContactMethodsNamePhone": "電話",
|
||||||
|
"accountContactMethodsNameAddress": "地址",
|
||||||
|
"accountContactMethodsPrimary": "主要的",
|
||||||
|
"accountContactMethodsVerified": "已驗證",
|
||||||
|
"accountContactMethodsPublic": "公開的",
|
||||||
|
"accountContactMethodsAdd": "添加聯繫方式",
|
||||||
|
"accountContactMethodsEdit": "編輯聯繫方式",
|
||||||
|
"accountContactMethodsAddDescription": "添加新的聯繫方式。",
|
||||||
|
"fieldContactContent": "聯繫方式",
|
||||||
|
"accountContactMethodsPublicHint": "這個聯繫方式公開地顯示在個人資料中。"
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ import 'package:surface/screens/account.dart';
|
|||||||
import 'package:surface/screens/account/account_settings.dart';
|
import 'package:surface/screens/account/account_settings.dart';
|
||||||
import 'package:surface/screens/account/action_events.dart';
|
import 'package:surface/screens/account/action_events.dart';
|
||||||
import 'package:surface/screens/account/badges.dart';
|
import 'package:surface/screens/account/badges.dart';
|
||||||
|
import 'package:surface/screens/account/contact_methods.dart';
|
||||||
import 'package:surface/screens/account/factor_settings.dart';
|
import 'package:surface/screens/account/factor_settings.dart';
|
||||||
import 'package:surface/screens/account/keypairs.dart';
|
import 'package:surface/screens/account/keypairs.dart';
|
||||||
import 'package:surface/screens/account/profile_page.dart';
|
import 'package:surface/screens/account/profile_page.dart';
|
||||||
@ -126,6 +127,11 @@ final _appRoutes = [
|
|||||||
name: 'account',
|
name: 'account',
|
||||||
builder: (context, state) => const AccountScreen(),
|
builder: (context, state) => const AccountScreen(),
|
||||||
routes: [
|
routes: [
|
||||||
|
GoRoute(
|
||||||
|
path: '/contacts',
|
||||||
|
name: 'accountContactMethods',
|
||||||
|
builder: (context, state) => const AccountContactMethod(),
|
||||||
|
),
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/events',
|
path: '/events',
|
||||||
name: 'accountActionEvents',
|
name: 'accountActionEvents',
|
||||||
|
@ -87,6 +87,16 @@ class AccountSettingsScreen extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
ListTile(
|
||||||
|
title: Text('accountContactMethods').tr(),
|
||||||
|
subtitle: Text('accountContactMethodsDescription').tr(),
|
||||||
|
contentPadding: const EdgeInsets.symmetric(horizontal: 24),
|
||||||
|
leading: const Icon(Symbols.contacts),
|
||||||
|
trailing: const Icon(Symbols.chevron_right),
|
||||||
|
onTap: () {
|
||||||
|
GoRouter.of(context).pushNamed('accountContactMethods');
|
||||||
|
},
|
||||||
|
),
|
||||||
ListTile(
|
ListTile(
|
||||||
title: Text('accountProfileEdit').tr(),
|
title: Text('accountProfileEdit').tr(),
|
||||||
subtitle: Text('accountProfileEditSubtitle').tr(),
|
subtitle: Text('accountProfileEditSubtitle').tr(),
|
||||||
|
292
lib/screens/account/contact_methods.dart
Normal file
292
lib/screens/account/contact_methods.dart
Normal file
@ -0,0 +1,292 @@
|
|||||||
|
import 'package:collection/collection.dart';
|
||||||
|
import 'package:dio/dio.dart';
|
||||||
|
import 'package:dropdown_button2/dropdown_button2.dart';
|
||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:gap/gap.dart';
|
||||||
|
import 'package:material_symbols_icons/symbols.dart';
|
||||||
|
import 'package:provider/provider.dart';
|
||||||
|
import 'package:styled_widget/styled_widget.dart';
|
||||||
|
import 'package:surface/providers/sn_network.dart';
|
||||||
|
import 'package:surface/types/account.dart';
|
||||||
|
import 'package:surface/widgets/dialog.dart';
|
||||||
|
import 'package:surface/widgets/loading_indicator.dart';
|
||||||
|
import 'package:surface/widgets/navigation/app_scaffold.dart';
|
||||||
|
|
||||||
|
const kContactMethodsIcons = [Symbols.email, Symbols.phone, Symbols.map];
|
||||||
|
const kContactMethodsName = ['Email', 'Phone', 'Address'];
|
||||||
|
|
||||||
|
class AccountContactMethod extends StatefulWidget {
|
||||||
|
const AccountContactMethod({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<AccountContactMethod> createState() => _AccountContactMethodState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _AccountContactMethodState extends State<AccountContactMethod> {
|
||||||
|
bool _isBusy = false;
|
||||||
|
List<SnAccountContact> _contactMethods = List.empty(growable: true);
|
||||||
|
|
||||||
|
Future<void> _fetchContactMethods() async {
|
||||||
|
setState(() => _isBusy = true);
|
||||||
|
try {
|
||||||
|
final sn = context.read<SnNetworkProvider>();
|
||||||
|
final resp = await sn.client.get('/cgi/id/users/me/contacts');
|
||||||
|
_contactMethods = List.from((resp.data as List<dynamic>)
|
||||||
|
.map((e) => SnAccountContact.fromJson(e)));
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
} finally {
|
||||||
|
setState(() => _isBusy = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_fetchContactMethods();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return AppScaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
leading: const PageBackButton(),
|
||||||
|
title: Text('accountContactMethods').tr(),
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
|
children: [
|
||||||
|
LoadingIndicator(isActive: _isBusy),
|
||||||
|
ListTile(
|
||||||
|
title: Text('accountContactMethodsAdd').tr(),
|
||||||
|
subtitle: Text('accountContactMethodsAddDescription').tr(),
|
||||||
|
contentPadding: const EdgeInsets.symmetric(horizontal: 24),
|
||||||
|
leading: const Icon(Symbols.add),
|
||||||
|
trailing: const Icon(Symbols.chevron_right),
|
||||||
|
onTap: () {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => _ContactMethodEditor(),
|
||||||
|
).then((value) {
|
||||||
|
if (value) {
|
||||||
|
_fetchContactMethods();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Divider(height: 1),
|
||||||
|
Expanded(
|
||||||
|
child: RefreshIndicator(
|
||||||
|
onRefresh: _fetchContactMethods,
|
||||||
|
child: ListView.builder(
|
||||||
|
padding: EdgeInsets.zero,
|
||||||
|
itemCount: _contactMethods.length,
|
||||||
|
itemBuilder: (context, index) {
|
||||||
|
final method = _contactMethods[index];
|
||||||
|
return ListTile(
|
||||||
|
title: Text(method.content),
|
||||||
|
subtitle: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
'accountContactMethodsName${kContactMethodsName[method.type]}',
|
||||||
|
).tr().bold(),
|
||||||
|
if (method.isPrimary ||
|
||||||
|
method.isPublic ||
|
||||||
|
method.verifiedAt != null)
|
||||||
|
Row(
|
||||||
|
spacing: 4,
|
||||||
|
children: [
|
||||||
|
if (method.isPrimary)
|
||||||
|
Text('accountContactMethodsPrimary').tr(),
|
||||||
|
if (method.isPublic)
|
||||||
|
Text('accountContactMethodsPublic').tr(),
|
||||||
|
if (method.verifiedAt != null)
|
||||||
|
Text('accountContactMethodsVerified').tr(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
contentPadding: const EdgeInsets.symmetric(horizontal: 24),
|
||||||
|
leading: Icon(
|
||||||
|
kContactMethodsIcons[method.type],
|
||||||
|
),
|
||||||
|
trailing: PopupMenuButton(
|
||||||
|
itemBuilder: (_) => [
|
||||||
|
PopupMenuItem(
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
const Icon(Symbols.edit),
|
||||||
|
const Gap(8),
|
||||||
|
Text('edit').tr(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
onTap: () {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => _ContactMethodEditor(
|
||||||
|
contact: method,
|
||||||
|
),
|
||||||
|
).then((value) {
|
||||||
|
if (value) {
|
||||||
|
_fetchContactMethods();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ContactMethodEditor extends StatefulWidget {
|
||||||
|
final SnAccountContact? contact;
|
||||||
|
const _ContactMethodEditor({this.contact});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<_ContactMethodEditor> createState() => _ContactMethodEditorState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ContactMethodEditorState extends State<_ContactMethodEditor> {
|
||||||
|
int _type = 0;
|
||||||
|
bool _isPublic = false;
|
||||||
|
final TextEditingController _contentController = TextEditingController();
|
||||||
|
|
||||||
|
bool _isBusy = false;
|
||||||
|
|
||||||
|
Future<void> _saveContactMethod() async {
|
||||||
|
setState(() => _isBusy = true);
|
||||||
|
try {
|
||||||
|
final sn = context.read<SnNetworkProvider>();
|
||||||
|
await sn.client.request(
|
||||||
|
widget.contact == null
|
||||||
|
? '/cgi/id/users/me/contacts'
|
||||||
|
: '/cgi/id/users/me/contacts/${widget.contact!.id}',
|
||||||
|
data: {
|
||||||
|
'content': _contentController.text,
|
||||||
|
'type': _type,
|
||||||
|
'is_public': _isPublic,
|
||||||
|
},
|
||||||
|
options: Options(
|
||||||
|
method: widget.contact == null ? 'POST' : 'PUT',
|
||||||
|
),
|
||||||
|
);
|
||||||
|
if (!mounted) return;
|
||||||
|
Navigator.pop(context, true);
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
} finally {
|
||||||
|
setState(() => _isBusy = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
if (widget.contact != null) {
|
||||||
|
_type = widget.contact!.type;
|
||||||
|
_isPublic = widget.contact!.isPublic;
|
||||||
|
_contentController.text = widget.contact!.content;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: widget.contact == null
|
||||||
|
? Text('accountContactMethodsAdd').tr()
|
||||||
|
: Text('accountContactMethodsEdit').tr(),
|
||||||
|
content: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
DropdownButtonHideUnderline(
|
||||||
|
child: DropdownButton2<int>(
|
||||||
|
value: _type,
|
||||||
|
items: kContactMethodsName
|
||||||
|
.mapIndexed((idx, ele) => DropdownMenuItem<int>(
|
||||||
|
value: idx,
|
||||||
|
child: Text('accountContactMethodsName$ele').tr(),
|
||||||
|
))
|
||||||
|
.toList(),
|
||||||
|
buttonStyleData: ButtonStyleData(
|
||||||
|
height: 48,
|
||||||
|
width: double.infinity,
|
||||||
|
padding: const EdgeInsets.only(left: 14, right: 14),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
borderRadius: BorderRadius.circular(4),
|
||||||
|
border: Border.all(
|
||||||
|
color: Theme.of(context).dividerColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
menuItemStyleData: const MenuItemStyleData(
|
||||||
|
height: 48,
|
||||||
|
padding: EdgeInsets.only(left: 14, right: 14),
|
||||||
|
),
|
||||||
|
onChanged: (value) {
|
||||||
|
setState(() => _type = value ?? 0);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Gap(8),
|
||||||
|
TextField(
|
||||||
|
controller: _contentController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
isDense: true,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
labelText: 'fieldContactContent'.tr(),
|
||||||
|
),
|
||||||
|
onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(),
|
||||||
|
),
|
||||||
|
const Gap(8),
|
||||||
|
Card(
|
||||||
|
margin: EdgeInsets.zero,
|
||||||
|
child: CheckboxListTile(
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.all(
|
||||||
|
Radius.circular(8),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
title: Text('accountContactMethodsPublic').tr(),
|
||||||
|
subtitle: Text('accountContactMethodsPublicHint').tr(),
|
||||||
|
secondary: const Icon(Symbols.globe),
|
||||||
|
value: _isPublic,
|
||||||
|
onChanged: (value) {
|
||||||
|
setState(() => _isPublic = value ?? false);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: _isBusy
|
||||||
|
? null
|
||||||
|
: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
child: Text('dialogDismiss').tr(),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: _isBusy
|
||||||
|
? null
|
||||||
|
: () {
|
||||||
|
_saveContactMethod();
|
||||||
|
},
|
||||||
|
child: Text('dialogConfirm').tr(),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user