2022-11-29 15:33:16 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2024-11-26 02:27:22 +00:00
|
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
2022-11-29 15:33:16 +00:00
|
|
|
import 'package:go_router/go_router.dart';
|
|
|
|
|
2023-04-27 19:19:51 +00:00
|
|
|
import '../controls/html_text_viewer_control.dart';
|
2023-03-20 14:06:44 +00:00
|
|
|
import '../controls/login_aware_cached_network_image.dart';
|
2022-12-14 21:53:46 +00:00
|
|
|
import '../controls/padding.dart';
|
2022-11-29 15:33:16 +00:00
|
|
|
import '../globals.dart';
|
2024-12-07 02:33:51 +00:00
|
|
|
import '../models/auth/profile.dart';
|
2022-11-29 15:33:16 +00:00
|
|
|
import '../models/connection.dart';
|
2024-08-28 15:23:44 +00:00
|
|
|
import '../models/timeline_grouping_list_data.dart';
|
2024-12-10 11:54:45 +00:00
|
|
|
import '../riverpod_controllers/account_services.dart';
|
2024-12-07 02:33:51 +00:00
|
|
|
import '../riverpod_controllers/blocks_services.dart';
|
|
|
|
import '../riverpod_controllers/circles_repo_services.dart';
|
|
|
|
import '../riverpod_controllers/connection_manager_services.dart';
|
2022-11-29 15:33:16 +00:00
|
|
|
import '../routes.dart';
|
2022-12-14 21:53:46 +00:00
|
|
|
import '../utils/snackbar_builder.dart';
|
2022-11-29 15:33:16 +00:00
|
|
|
import '../utils/url_opening_utils.dart';
|
|
|
|
|
2024-11-26 02:27:22 +00:00
|
|
|
class UserProfileScreen extends ConsumerStatefulWidget {
|
2022-11-29 15:33:16 +00:00
|
|
|
final String userId;
|
|
|
|
|
|
|
|
const UserProfileScreen({super.key, required this.userId});
|
|
|
|
|
2022-12-14 21:53:46 +00:00
|
|
|
@override
|
2024-11-26 02:27:22 +00:00
|
|
|
ConsumerState<UserProfileScreen> createState() => _UserProfileScreenState();
|
2022-12-14 21:53:46 +00:00
|
|
|
}
|
|
|
|
|
2024-11-26 02:27:22 +00:00
|
|
|
class _UserProfileScreenState extends ConsumerState<UserProfileScreen> {
|
2022-12-14 21:53:46 +00:00
|
|
|
var isUpdating = false;
|
|
|
|
|
2023-04-28 01:48:01 +00:00
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
2024-12-10 11:54:45 +00:00
|
|
|
final profile = ref.read(activeProfileProvider);
|
2024-12-07 02:33:51 +00:00
|
|
|
ref.read(connectionByIdProvider(profile, widget.userId, forceUpdate: true));
|
2023-04-28 01:48:01 +00:00
|
|
|
}
|
|
|
|
|
2022-11-29 15:33:16 +00:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-12-10 11:54:45 +00:00
|
|
|
final profile = ref.watch(activeProfileProvider);
|
2024-12-20 14:45:39 +00:00
|
|
|
ref.watch(blocksManagerProvider(profile));
|
2024-12-07 02:33:51 +00:00
|
|
|
final body = ref.watch(connectionByIdProvider(profile, widget.userId)).fold(
|
|
|
|
onSuccess: (connectionProfile) {
|
2024-12-10 11:54:45 +00:00
|
|
|
final notMyProfile = profile.userId != connectionProfile.id;
|
2022-12-14 21:53:46 +00:00
|
|
|
|
|
|
|
return RefreshIndicator(
|
|
|
|
onRefresh: () async {
|
2024-12-07 02:33:51 +00:00
|
|
|
await ref
|
2024-12-20 14:45:39 +00:00
|
|
|
.read(blocksManagerProvider(profile).notifier)
|
|
|
|
.updateBlock(connectionProfile);
|
2024-06-28 13:39:18 +00:00
|
|
|
setState(() {});
|
2022-12-14 21:53:46 +00:00
|
|
|
},
|
|
|
|
child: SingleChildScrollView(
|
|
|
|
physics: const AlwaysScrollableScrollPhysics(),
|
|
|
|
child: Column(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.start,
|
2022-12-14 22:07:18 +00:00
|
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
2022-12-14 21:53:46 +00:00
|
|
|
children: [
|
2023-03-20 14:06:44 +00:00
|
|
|
LoginAwareCachedNetworkImage(
|
2024-12-07 02:33:51 +00:00
|
|
|
imageUrl: connectionProfile.avatarUrl.toString()),
|
2023-10-31 01:44:16 +00:00
|
|
|
const Row(
|
2022-12-14 22:07:18 +00:00
|
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
2022-12-28 20:56:27 +00:00
|
|
|
children: [],
|
|
|
|
),
|
|
|
|
Text(
|
|
|
|
notMyProfile
|
2024-12-07 02:33:51 +00:00
|
|
|
? '${connectionProfile.name} (${connectionProfile.handle})'
|
|
|
|
: '${connectionProfile.name} (Your Account)',
|
2022-12-28 20:56:27 +00:00
|
|
|
softWrap: true,
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
style: Theme.of(context).textTheme.titleLarge,
|
2022-12-14 21:53:46 +00:00
|
|
|
),
|
2022-12-14 22:07:18 +00:00
|
|
|
const VerticalPadding(),
|
2024-12-17 02:46:51 +00:00
|
|
|
if (connectionProfile.status != ConnectionStatus.unknown)
|
|
|
|
Text('( ${connectionProfile.status.label()} )'),
|
2022-12-28 20:56:27 +00:00
|
|
|
const VerticalPadding(),
|
2023-05-03 19:49:40 +00:00
|
|
|
Wrap(
|
|
|
|
spacing: 10.0,
|
|
|
|
runSpacing: 10.0,
|
2022-12-14 22:07:18 +00:00
|
|
|
children: [
|
|
|
|
ElevatedButton(
|
2024-12-24 18:39:33 +00:00
|
|
|
onPressed: () => context.pushNamed(
|
|
|
|
ScreenPaths.userPosts,
|
|
|
|
pathParameters: {'id': connectionProfile.id},
|
|
|
|
),
|
|
|
|
child: const Text('Posts'),
|
|
|
|
),
|
2024-12-24 19:04:56 +00:00
|
|
|
ElevatedButton(
|
|
|
|
onPressed: () => context.pushNamed(
|
|
|
|
ScreenPaths.userPostsAndComments,
|
|
|
|
pathParameters: {'id': connectionProfile.id},
|
|
|
|
),
|
|
|
|
child: const Text('Posts & Comments'),
|
|
|
|
),
|
2024-12-24 18:39:33 +00:00
|
|
|
ElevatedButton(
|
|
|
|
onPressed: () => context.pushNamed(
|
|
|
|
ScreenPaths.userMedia,
|
|
|
|
pathParameters: {'id': connectionProfile.id},
|
|
|
|
),
|
|
|
|
child: const Text('Media'),
|
|
|
|
),
|
2022-12-14 22:07:18 +00:00
|
|
|
ElevatedButton(
|
|
|
|
onPressed: () async =>
|
2024-12-07 02:33:51 +00:00
|
|
|
await openProfileExternal(context, connectionProfile),
|
2022-12-14 22:07:18 +00:00
|
|
|
child: const Text('Open In Browser'),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
const VerticalPadding(),
|
2023-11-16 21:24:48 +00:00
|
|
|
Wrap(
|
|
|
|
spacing: 10.0,
|
|
|
|
runSpacing: 10.0,
|
|
|
|
children: [
|
|
|
|
if (notMyProfile) ...[
|
|
|
|
buildConnectionStatusToggle(
|
2024-12-07 02:33:51 +00:00
|
|
|
context,
|
|
|
|
profile,
|
|
|
|
connectionProfile,
|
|
|
|
),
|
2024-12-20 14:45:39 +00:00
|
|
|
buildBlockToggle(context, profile, connectionProfile),
|
2023-11-16 21:24:48 +00:00
|
|
|
],
|
|
|
|
],
|
|
|
|
),
|
|
|
|
const VerticalPadding(),
|
2023-04-27 19:19:51 +00:00
|
|
|
HtmlTextViewerControl(
|
2024-12-07 02:33:51 +00:00
|
|
|
content: connectionProfile.note,
|
2023-01-31 20:27:26 +00:00
|
|
|
onTapUrl: (url) async {
|
|
|
|
return await openUrlStringInSystembrowser(
|
|
|
|
context, url, 'link');
|
|
|
|
},
|
|
|
|
),
|
|
|
|
const VerticalPadding(),
|
|
|
|
Text(
|
2024-12-07 02:33:51 +00:00
|
|
|
'#Followers: ${connectionProfile.followerCount} followers, #Following, ${connectionProfile.followingCount}, #Statuses: ${connectionProfile.statusesCount}'),
|
2023-01-31 20:27:26 +00:00
|
|
|
const VerticalPadding(),
|
2024-12-07 02:33:51 +00:00
|
|
|
Text('Last Status: ${connectionProfile.lastStatus ?? "Unknown"}'),
|
2023-01-31 20:27:26 +00:00
|
|
|
const VerticalPadding(),
|
2024-12-07 02:33:51 +00:00
|
|
|
if (connectionProfile.status == ConnectionStatus.mutual ||
|
|
|
|
connectionProfile.status == ConnectionStatus.youFollowThem)
|
|
|
|
buildCircles(context, profile, connectionProfile),
|
2022-12-14 21:53:46 +00:00
|
|
|
],
|
2022-11-29 15:33:16 +00:00
|
|
|
),
|
2022-12-14 21:53:46 +00:00
|
|
|
),
|
2022-11-29 15:33:16 +00:00
|
|
|
);
|
|
|
|
}, onError: (error) {
|
|
|
|
return Text('Error getting profile: $error');
|
|
|
|
});
|
|
|
|
return Scaffold(
|
|
|
|
appBar: AppBar(
|
2023-10-31 01:44:16 +00:00
|
|
|
title: const Text('Profile'),
|
2024-12-17 03:47:44 +00:00
|
|
|
leading: context.canPop()
|
|
|
|
? null
|
|
|
|
: IconButton(
|
|
|
|
onPressed: () => context.go(ScreenPaths.timelines),
|
|
|
|
icon: const Icon(Icons.arrow_back),
|
|
|
|
),
|
2022-11-29 15:33:16 +00:00
|
|
|
),
|
|
|
|
body: Padding(
|
|
|
|
padding: const EdgeInsets.all(8.0),
|
|
|
|
child: Center(
|
|
|
|
child: body,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
2022-12-14 15:50:17 +00:00
|
|
|
|
2023-11-15 21:05:45 +00:00
|
|
|
Widget buildCircles(
|
2022-12-14 15:50:17 +00:00
|
|
|
BuildContext context,
|
2024-12-07 02:33:51 +00:00
|
|
|
Profile profile,
|
|
|
|
Connection connectionProfile,
|
2022-12-14 15:50:17 +00:00
|
|
|
) {
|
2024-12-07 02:33:51 +00:00
|
|
|
final myCircles =
|
|
|
|
ref.watch(timelineGroupingListProvider(profile, GroupingType.circle));
|
|
|
|
final usersCircles = ref
|
|
|
|
.watch(circlesProvider(profile).notifier)
|
|
|
|
.getCirclesForUser(connectionProfile.id)
|
|
|
|
.fold(
|
|
|
|
onSuccess: (circles) => circles.toSet(),
|
|
|
|
onError: (error) {
|
|
|
|
buildSnackbar(context, 'Error getting circle data: $error');
|
|
|
|
return <TimelineGroupingListData>{};
|
|
|
|
});
|
|
|
|
if (myCircles.isNotEmpty) {
|
|
|
|
myCircles.sort((g1, g2) => g1.name.compareTo(g2.name));
|
|
|
|
}
|
2024-06-28 14:12:14 +00:00
|
|
|
|
2023-11-15 21:05:45 +00:00
|
|
|
final circlesWidgets = myCircles.map((g) {
|
2022-12-14 21:53:46 +00:00
|
|
|
return CheckboxListTile(
|
|
|
|
title: Text(g.name),
|
2023-11-15 21:05:45 +00:00
|
|
|
value: usersCircles.contains(g),
|
2024-06-28 14:12:14 +00:00
|
|
|
onChanged: isUpdating
|
|
|
|
? null
|
|
|
|
: (bool? value) async {
|
|
|
|
if (isUpdating) {
|
|
|
|
return;
|
|
|
|
}
|
2023-11-15 20:21:10 +00:00
|
|
|
|
2024-06-28 14:12:14 +00:00
|
|
|
final isAdding = value == true;
|
|
|
|
final confirm = await showYesNoDialog(
|
|
|
|
context,
|
|
|
|
isAdding
|
|
|
|
? 'Add user to ${g.name}'
|
|
|
|
: 'Remove user from ${g.name}');
|
|
|
|
if (confirm != true) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
isUpdating = true;
|
|
|
|
});
|
|
|
|
if (isAdding) {
|
2024-12-20 14:15:30 +00:00
|
|
|
await ref
|
2024-12-07 02:33:51 +00:00
|
|
|
.watch(circlesProvider(profile).notifier)
|
|
|
|
.addConnectionToCircle(g, connectionProfile);
|
2024-06-28 14:12:14 +00:00
|
|
|
} else {
|
2024-12-07 02:33:51 +00:00
|
|
|
await ref
|
|
|
|
.watch(circlesProvider(profile).notifier)
|
|
|
|
.removeConnectionFromCircle(g, connectionProfile);
|
2024-06-28 14:12:14 +00:00
|
|
|
}
|
2024-07-26 14:15:24 +00:00
|
|
|
if (context.mounted) {
|
2024-06-28 14:12:14 +00:00
|
|
|
buildSnackbar(context, "User's Circles Updated");
|
|
|
|
}
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
isUpdating = false;
|
|
|
|
});
|
|
|
|
},
|
2022-12-14 21:53:46 +00:00
|
|
|
);
|
|
|
|
}).toList();
|
|
|
|
return Column(
|
|
|
|
children: [
|
|
|
|
Text(
|
2023-11-15 21:05:45 +00:00
|
|
|
'Circles: ',
|
2022-12-14 21:53:46 +00:00
|
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
|
|
),
|
|
|
|
const VerticalPadding(),
|
2023-11-15 21:05:45 +00:00
|
|
|
...circlesWidgets,
|
2022-12-14 21:53:46 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
2022-12-14 15:50:17 +00:00
|
|
|
|
2023-05-03 19:49:40 +00:00
|
|
|
Widget buildBlockToggle(
|
|
|
|
BuildContext context,
|
2024-12-20 14:45:39 +00:00
|
|
|
Profile profile,
|
|
|
|
Connection connection,
|
2023-05-03 19:49:40 +00:00
|
|
|
) {
|
|
|
|
late Widget blockToggleButton;
|
2024-12-20 14:45:39 +00:00
|
|
|
switch (connection.status) {
|
2023-05-03 19:49:40 +00:00
|
|
|
case ConnectionStatus.blocked:
|
|
|
|
blockToggleButton = ElevatedButton(
|
|
|
|
onPressed: isUpdating
|
|
|
|
? null
|
|
|
|
: () async {
|
2024-12-20 14:45:39 +00:00
|
|
|
final confirm = await showYesNoDialog(
|
|
|
|
context, 'Unblock ${connection.name}');
|
2023-05-03 19:49:40 +00:00
|
|
|
if (confirm != true) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
isUpdating = true;
|
|
|
|
});
|
2024-12-20 14:45:39 +00:00
|
|
|
await ref
|
|
|
|
.read(blocksManagerProvider(profile).notifier)
|
|
|
|
.unblockConnection(connection);
|
2023-05-03 19:49:40 +00:00
|
|
|
setState(() {
|
|
|
|
isUpdating = false;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
child: const Text('Unblock'),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ConnectionStatus.mutual:
|
|
|
|
case ConnectionStatus.theyFollowYou:
|
|
|
|
case ConnectionStatus.youFollowThem:
|
|
|
|
case ConnectionStatus.none:
|
|
|
|
case ConnectionStatus.unknown:
|
|
|
|
blockToggleButton = ElevatedButton(
|
|
|
|
onPressed: isUpdating
|
|
|
|
? null
|
|
|
|
: () async {
|
2024-12-20 14:45:39 +00:00
|
|
|
final confirm = await showYesNoDialog(
|
|
|
|
context, 'Block ${connection.name}');
|
2023-05-03 19:49:40 +00:00
|
|
|
if (confirm != true) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
isUpdating = true;
|
|
|
|
});
|
2024-12-20 14:45:39 +00:00
|
|
|
await ref
|
|
|
|
.read(blocksManagerProvider(profile).notifier)
|
|
|
|
.blockConnection(connection);
|
2023-05-03 19:49:40 +00:00
|
|
|
setState(() {
|
|
|
|
isUpdating = false;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
child: const Text('Block'),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ConnectionStatus.you:
|
|
|
|
blockToggleButton = const SizedBox();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return blockToggleButton;
|
|
|
|
}
|
|
|
|
|
2022-12-14 22:07:18 +00:00
|
|
|
Widget buildConnectionStatusToggle(
|
2022-12-14 21:53:46 +00:00
|
|
|
BuildContext context,
|
2024-12-07 02:33:51 +00:00
|
|
|
Profile profile,
|
|
|
|
Connection connectionProfile,
|
2022-12-14 21:53:46 +00:00
|
|
|
) {
|
2022-12-14 15:50:17 +00:00
|
|
|
late Widget followToggleButton;
|
2024-12-07 02:33:51 +00:00
|
|
|
switch (connectionProfile.status) {
|
2022-12-14 15:50:17 +00:00
|
|
|
case ConnectionStatus.mutual:
|
|
|
|
case ConnectionStatus.youFollowThem:
|
|
|
|
followToggleButton = ElevatedButton(
|
2022-12-14 21:53:46 +00:00
|
|
|
onPressed: isUpdating
|
|
|
|
? null
|
|
|
|
: () async {
|
2022-12-29 05:31:10 +00:00
|
|
|
final confirm = await showYesNoDialog(
|
2024-12-07 02:33:51 +00:00
|
|
|
context, 'Unfollow ${connectionProfile.name}');
|
2022-12-29 05:31:10 +00:00
|
|
|
if (confirm != true) {
|
|
|
|
return;
|
|
|
|
}
|
2022-12-14 21:53:46 +00:00
|
|
|
setState(() {
|
|
|
|
isUpdating = true;
|
|
|
|
});
|
2024-12-07 02:33:51 +00:00
|
|
|
await ref
|
|
|
|
.read(
|
|
|
|
connectionModifierProvider(profile, connectionProfile)
|
|
|
|
.notifier)
|
|
|
|
.unfollow();
|
2022-12-14 21:53:46 +00:00
|
|
|
setState(() {
|
|
|
|
isUpdating = false;
|
|
|
|
});
|
|
|
|
},
|
2022-12-14 15:50:17 +00:00
|
|
|
child: const Text('Unfollow'),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ConnectionStatus.theyFollowYou:
|
|
|
|
case ConnectionStatus.none:
|
|
|
|
followToggleButton = ElevatedButton(
|
2022-12-14 21:53:46 +00:00
|
|
|
onPressed: isUpdating
|
|
|
|
? null
|
|
|
|
: () async {
|
2024-12-07 02:33:51 +00:00
|
|
|
final confirm = await showYesNoDialog(
|
|
|
|
context, 'Follow ${connectionProfile.name}');
|
2022-12-29 05:31:10 +00:00
|
|
|
if (confirm != true) {
|
|
|
|
return;
|
|
|
|
}
|
2022-12-14 21:53:46 +00:00
|
|
|
setState(() {
|
|
|
|
isUpdating = true;
|
|
|
|
});
|
2024-12-07 02:33:51 +00:00
|
|
|
await ref
|
|
|
|
.read(
|
|
|
|
connectionModifierProvider(profile, connectionProfile)
|
|
|
|
.notifier)
|
|
|
|
.follow();
|
2022-12-14 21:53:46 +00:00
|
|
|
setState(() {
|
|
|
|
isUpdating = false;
|
|
|
|
});
|
|
|
|
},
|
2022-12-14 15:50:17 +00:00
|
|
|
child: const Text('Follow'),
|
|
|
|
);
|
|
|
|
break;
|
2023-05-03 19:49:40 +00:00
|
|
|
case ConnectionStatus.blocked:
|
2022-12-14 15:50:17 +00:00
|
|
|
case ConnectionStatus.you:
|
|
|
|
case ConnectionStatus.unknown:
|
|
|
|
followToggleButton = const SizedBox();
|
|
|
|
break;
|
|
|
|
}
|
2022-12-14 22:07:18 +00:00
|
|
|
return followToggleButton;
|
2022-12-14 15:50:17 +00:00
|
|
|
}
|
2022-11-29 15:33:16 +00:00
|
|
|
}
|