Solian/lib/screens/realms/realm_view.dart

271 lines
7.8 KiB
Dart
Raw Normal View History

2024-05-29 22:42:11 +08:00
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:infinite_scroll_pagination/infinite_scroll_pagination.dart';
import 'package:solian/exts.dart';
2024-05-29 23:22:24 +08:00
import 'package:solian/models/channel.dart';
2024-05-29 22:42:11 +08:00
import 'package:solian/models/pagination.dart';
import 'package:solian/models/post.dart';
import 'package:solian/models/realm.dart';
2024-05-29 23:22:24 +08:00
import 'package:solian/providers/auth.dart';
import 'package:solian/providers/content/channel.dart';
import 'package:solian/providers/content/posts.dart';
2024-05-29 22:42:11 +08:00
import 'package:solian/providers/content/realm.dart';
import 'package:solian/router.dart';
2024-05-29 23:22:24 +08:00
import 'package:solian/screens/channel/channel_organize.dart';
2024-05-29 22:42:11 +08:00
import 'package:solian/theme.dart';
import 'package:solian/widgets/app_bar_leading.dart';
2024-05-29 23:22:24 +08:00
import 'package:solian/widgets/channel/channel_list.dart';
2024-05-29 22:42:11 +08:00
import 'package:solian/widgets/posts/post_list.dart';
class RealmViewScreen extends StatefulWidget {
final String alias;
const RealmViewScreen({super.key, required this.alias});
@override
State<RealmViewScreen> createState() => _RealmViewScreenState();
}
class _RealmViewScreenState extends State<RealmViewScreen> {
bool _isBusy = false;
String? _overrideAlias;
Realm? _realm;
2024-05-29 23:22:24 +08:00
final List<Channel> _channels = List.empty(growable: true);
2024-05-29 22:42:11 +08:00
getRealm({String? overrideAlias}) async {
final RealmProvider realm = Get.find();
2024-05-29 22:42:11 +08:00
setState(() => _isBusy = true);
if (overrideAlias != null) {
_overrideAlias = overrideAlias;
}
try {
final resp = await realm.getRealm(_overrideAlias ?? widget.alias);
2024-05-29 22:42:11 +08:00
setState(() => _realm = Realm.fromJson(resp.body));
} catch (e) {
context.showErrorDialog(e);
}
setState(() => _isBusy = false);
}
2024-05-29 23:22:24 +08:00
getChannels() async {
setState(() => _isBusy = true);
final ChannelProvider channel = Get.find();
final resp = await channel.listChannel(scope: _realm!.alias);
final availableResp = await channel.listAvailableChannel(
scope: _realm!.alias,
);
final Set<int> channelIdx = {};
2024-05-29 23:22:24 +08:00
setState(() {
_channels.clear();
_channels.addAll(
resp.body.map((e) => Channel.fromJson(e)).toList().cast<Channel>(),
);
_channels.addAll(
availableResp.body
.map((e) => Channel.fromJson(e))
.toList()
.cast<Channel>(),
);
_channels.retainWhere((x) => channelIdx.add(x.id));
2024-05-29 23:22:24 +08:00
});
setState(() => _isBusy = false);
}
2024-05-29 22:42:11 +08:00
@override
void initState() {
super.initState();
2024-05-29 23:22:24 +08:00
getRealm().then((_) {
getChannels();
});
2024-05-29 22:42:11 +08:00
}
@override
Widget build(BuildContext context) {
return Material(
color: Theme.of(context).colorScheme.surface,
child: DefaultTabController(
length: 2,
child: NestedScrollView(
headerSliverBuilder: (context, innerBoxIsScrolled) {
return [
SliverOverlapAbsorber(
handle:
NestedScrollView.sliverOverlapAbsorberHandleFor(context),
sliver: SliverAppBar(
2024-07-12 22:37:58 +08:00
leading: AppBarLeadingButton.adaptive(context),
title: Text(_realm?.name ?? 'loading'.tr),
centerTitle: false,
actions: [
IconButton(
icon: const Icon(Icons.more_vert),
onPressed: () {
AppRouter.instance
.pushNamed(
'realmDetail',
pathParameters: {'alias': widget.alias},
extra: _realm,
)
.then((value) {
if (value == false) AppRouter.instance.pop();
if (value != null) {
final resp =
Realm.fromJson(value as Map<String, dynamic>);
getRealm(overrideAlias: resp.alias);
}
});
},
2024-05-29 22:42:11 +08:00
),
SizedBox(
width: AppTheme.isLargeScreen(context) ? 8 : 16,
2024-05-29 23:22:24 +08:00
),
2024-05-29 22:42:11 +08:00
],
bottom: const TabBar(
isScrollable: true,
tabs: [
Tab(icon: Icon(Icons.feed)),
Tab(icon: Icon(Icons.chat)),
],
),
),
)
];
},
body: Builder(
builder: (context) {
if (_isBusy) {
return const Center(child: CircularProgressIndicator());
}
return TabBarView(
physics: const NeverScrollableScrollPhysics(),
children: [
RealmPostListWidget(realm: _realm!),
RealmChannelListWidget(
realm: _realm!,
channels: _channels,
onRefresh: () => getChannels(),
),
],
);
},
2024-05-29 22:42:11 +08:00
),
),
),
);
}
}
class RealmPostListWidget extends StatefulWidget {
final Realm realm;
const RealmPostListWidget({super.key, required this.realm});
@override
State<RealmPostListWidget> createState() => _RealmPostListWidgetState();
}
class _RealmPostListWidgetState extends State<RealmPostListWidget> {
final PagingController<int, Post> _pagingController =
PagingController(firstPageKey: 0);
getPosts(int pageKey) async {
final PostProvider provider = Get.find();
2024-05-29 22:42:11 +08:00
Response resp;
try {
resp = await provider.listPost(pageKey, realm: widget.realm.alias);
2024-05-29 22:42:11 +08:00
} catch (e) {
_pagingController.error = e;
return;
}
final PaginationResult result = PaginationResult.fromJson(resp.body);
final parsed = result.data?.map((e) => Post.fromJson(e)).toList();
if (parsed != null && parsed.length >= 10) {
_pagingController.appendPage(parsed, pageKey + parsed.length);
} else if (parsed != null) {
_pagingController.appendLastPage(parsed);
}
}
@override
void initState() {
super.initState();
_pagingController.addPageRequestListener(getPosts);
}
@override
Widget build(BuildContext context) {
return RefreshIndicator(
onRefresh: () => Future.sync(() => _pagingController.refresh()),
child: CustomScrollView(
slivers: [
PostListWidget(controller: _pagingController),
],
),
);
}
}
2024-05-29 23:22:24 +08:00
class RealmChannelListWidget extends StatelessWidget {
final Realm realm;
final List<Channel> channels;
final Future Function() onRefresh;
const RealmChannelListWidget({
super.key,
required this.realm,
required this.channels,
required this.onRefresh,
});
@override
Widget build(BuildContext context) {
final AuthProvider auth = Get.find();
2024-07-25 01:18:47 +08:00
return RefreshIndicator(
onRefresh: onRefresh,
child: Column(
children: [
ListTile(
leading: const Icon(Icons.add_box),
contentPadding: const EdgeInsets.only(left: 32, right: 8),
tileColor: Theme.of(context).colorScheme.surfaceContainer,
title: Text('channelNew'.tr),
subtitle: Text(
'channelNewInRealmHint'.trParams({'realm': '#${realm.alias}'}),
),
onTap: () {
AppRouter.instance
.pushNamed(
'channelOrganizing',
extra: ChannelOrganizeArguments(realm: realm),
2024-05-29 23:22:24 +08:00
)
2024-07-25 01:18:47 +08:00
.then((value) {
if (value != null) onRefresh();
});
},
2024-05-29 23:22:24 +08:00
),
2024-07-25 01:18:47 +08:00
Expanded(
child: ChannelListWidget(
channels: channels,
selfId: auth.userProfile.value!['id'],
noCategory: true,
),
)
],
),
2024-05-29 23:22:24 +08:00
);
}
}