Solian/lib/widgets/posts/reaction_action.dart

146 lines
3.7 KiB
Dart
Raw Normal View History

2024-04-15 23:08:32 +08:00
import 'dart:convert';
import 'package:flutter/material.dart';
2024-04-25 21:33:53 +08:00
import 'package:flutter_animate/flutter_animate.dart';
2024-04-15 23:08:32 +08:00
import 'package:provider/provider.dart';
import 'package:solian/models/reaction.dart';
import 'package:solian/providers/auth.dart';
import 'package:solian/utils/service_url.dart';
2024-04-15 23:40:36 +08:00
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
2024-04-29 20:22:06 +08:00
import 'package:solian/widgets/exts.dart';
2024-04-15 23:08:32 +08:00
Future<void> doReact(
String dataset,
int id,
String symbol,
int attitude,
final void Function(String symbol, int num) onReact,
BuildContext context,
) async {
final auth = context.read<AuthProvider>();
if (!await auth.isAuthorized()) return;
var uri = getRequestUri(
'interactive',
'/api/p/$dataset/$id/react',
);
var res = await auth.client!.post(
uri,
headers: <String, String>{
'Content-Type': 'application/json',
},
body: jsonEncode(<String, dynamic>{
'symbol': symbol,
'attitude': attitude,
}),
);
if (res.statusCode == 201) {
onReact(symbol, 1);
ScaffoldMessenger.of(context).showSnackBar(
2024-04-15 23:40:36 +08:00
SnackBar(
content: Text(AppLocalizations.of(context)!.reactionAdded),
2024-04-15 23:08:32 +08:00
),
);
} else if (res.statusCode == 204) {
onReact(symbol, -1);
ScaffoldMessenger.of(context).showSnackBar(
2024-04-15 23:40:36 +08:00
SnackBar(
content: Text(AppLocalizations.of(context)!.reactionRemoved),
2024-04-15 23:08:32 +08:00
),
);
} else {
2024-04-15 23:40:36 +08:00
final message = utf8.decode(res.bodyBytes);
2024-04-29 20:22:06 +08:00
context.showErrorDialog(message);
2024-04-15 23:08:32 +08:00
}
if (Navigator.canPop(context)) {
Navigator.pop(context);
}
}
class ReactionActionPopup extends StatefulWidget {
final String dataset;
final int id;
final void Function(String symbol, int num) onReact;
const ReactionActionPopup({
super.key,
required this.dataset,
required this.id,
required this.onReact,
});
@override
State<ReactionActionPopup> createState() => _ReactionActionPopupState();
}
class _ReactionActionPopupState extends State<ReactionActionPopup> {
bool _isSubmitting = false;
Future<void> doWidgetReact(
String symbol,
int attitude,
BuildContext context,
) async {
if (_isSubmitting) return;
setState(() => _isSubmitting = true);
await doReact(
widget.dataset,
widget.id,
symbol,
attitude,
widget.onReact,
context,
);
setState(() => _isSubmitting = false);
}
@override
Widget build(BuildContext context) {
final reactEntries = reactions.entries.toList();
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Container(
2024-04-25 21:33:53 +08:00
padding: const EdgeInsets.only(left: 8, right: 8, top: 20, bottom: 12),
2024-04-15 23:08:32 +08:00
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: 8,
vertical: 12,
),
child: Text(
2024-04-15 23:40:36 +08:00
AppLocalizations.of(context)!.reaction,
2024-04-15 23:08:32 +08:00
style: Theme.of(context).textTheme.headlineSmall,
),
),
),
2024-04-25 21:33:53 +08:00
_isSubmitting ? const LinearProgressIndicator().animate().scaleX() : Container(),
2024-04-15 23:08:32 +08:00
Expanded(
child: ListView.builder(
itemCount: reactions.length,
itemBuilder: (BuildContext context, int index) {
var info = reactEntries[index];
return InkWell(
onTap: () async {
await doWidgetReact(info.key, info.value.attitude, context);
},
child: ListTile(
title: Text(info.value.icon),
subtitle: Text(
":${info.key}:",
style: const TextStyle(fontFamily: "monospace"),
),
),
);
},
),
),
],
);
}
}