mirror of
https://gitlab.com/famedly/fluffychat.git
synced 2025-01-12 10:42:35 +01:00
251 lines
7.5 KiB
Dart
251 lines
7.5 KiB
Dart
import 'dart:async';
|
|
import 'dart:io';
|
|
|
|
import 'package:adaptive_dialog/adaptive_dialog.dart';
|
|
import 'package:adaptive_page_layout/adaptive_page_layout.dart';
|
|
import 'package:famedlysdk/famedlysdk.dart';
|
|
import 'package:fluffychat/views/home_view_parts/discover.dart';
|
|
import 'package:fluffychat/views/share_view.dart';
|
|
import 'package:flutter/cupertino.dart';
|
|
import 'package:fluffychat/app_config.dart';
|
|
import 'package:fluffychat/utils/platform_infos.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:preload_page_view/preload_page_view.dart';
|
|
import 'package:receive_sharing_intent/receive_sharing_intent.dart';
|
|
import '../components/matrix.dart';
|
|
import '../utils/matrix_file_extension.dart';
|
|
import '../utils/url_launcher.dart';
|
|
import 'home_view_parts/chat_list.dart';
|
|
import 'home_view_parts/status_list.dart';
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
|
|
enum SelectMode { normal, share, select }
|
|
|
|
class HomeView extends StatefulWidget {
|
|
final String activeChat;
|
|
|
|
const HomeView({this.activeChat, Key key}) : super(key: key);
|
|
|
|
@override
|
|
_HomeViewState createState() => _HomeViewState();
|
|
}
|
|
|
|
class _HomeViewState extends State<HomeView> {
|
|
@override
|
|
void initState() {
|
|
_initReceiveSharingIntent();
|
|
super.initState();
|
|
}
|
|
|
|
int currentIndex = 1;
|
|
|
|
StreamSubscription _intentDataStreamSubscription;
|
|
|
|
StreamSubscription _intentFileStreamSubscription;
|
|
|
|
StreamSubscription _onShareContentChanged;
|
|
|
|
AppBar appBar;
|
|
|
|
final PreloadPageController _pageController =
|
|
PreloadPageController(initialPage: 1);
|
|
|
|
void _onShare(Map<String, dynamic> content) {
|
|
if (content != null) {
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => Navigator.of(context).push(
|
|
MaterialPageRoute(
|
|
builder: (_) => ShareView(),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
void _processIncomingSharedFiles(List<SharedMediaFile> files) {
|
|
if (files?.isEmpty ?? true) return;
|
|
AdaptivePageLayout.of(context).popUntilIsFirst();
|
|
final file = File(files.first.path);
|
|
|
|
Matrix.of(context).shareContent = {
|
|
'msgtype': 'chat.fluffy.shared_file',
|
|
'file': MatrixFile(
|
|
bytes: file.readAsBytesSync(),
|
|
name: file.path,
|
|
).detectFileType,
|
|
};
|
|
}
|
|
|
|
void _processIncomingSharedText(String text) {
|
|
if (text == null) return;
|
|
AdaptivePageLayout.of(context).popUntilIsFirst();
|
|
if (text.toLowerCase().startsWith(AppConfig.inviteLinkPrefix) ||
|
|
(text.toLowerCase().startsWith(AppConfig.schemePrefix) &&
|
|
!RegExp(r'\s').hasMatch(text))) {
|
|
UrlLauncher(context, text).openMatrixToUrl();
|
|
return;
|
|
}
|
|
Matrix.of(context).shareContent = {
|
|
'msgtype': 'm.text',
|
|
'body': text,
|
|
};
|
|
}
|
|
|
|
void _initReceiveSharingIntent() {
|
|
if (!PlatformInfos.isMobile) return;
|
|
|
|
// For sharing images coming from outside the app while the app is in the memory
|
|
_intentFileStreamSubscription = ReceiveSharingIntent.getMediaStream()
|
|
.listen(_processIncomingSharedFiles, onError: print);
|
|
|
|
// For sharing images coming from outside the app while the app is closed
|
|
ReceiveSharingIntent.getInitialMedia().then(_processIncomingSharedFiles);
|
|
|
|
// For sharing or opening urls/text coming from outside the app while the app is in the memory
|
|
_intentDataStreamSubscription = ReceiveSharingIntent.getTextStream()
|
|
.listen(_processIncomingSharedText, onError: print);
|
|
|
|
// For sharing or opening urls/text coming from outside the app while the app is closed
|
|
ReceiveSharingIntent.getInitialText().then(_processIncomingSharedText);
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_intentDataStreamSubscription?.cancel();
|
|
_intentFileStreamSubscription?.cancel();
|
|
super.dispose();
|
|
}
|
|
|
|
String _server;
|
|
|
|
void _setServer(BuildContext context) async {
|
|
final newServer = await showTextInputDialog(
|
|
title: L10n.of(context).changeTheHomeserver,
|
|
context: context,
|
|
textFields: [
|
|
DialogTextField(
|
|
prefixText: 'https://',
|
|
hintText: Matrix.of(context).client.homeserver.host,
|
|
initialText: _server,
|
|
keyboardType: TextInputType.url,
|
|
)
|
|
]);
|
|
if (newServer == null) return;
|
|
setState(() {
|
|
_server = newServer.single;
|
|
});
|
|
}
|
|
|
|
void _onFabTab() {
|
|
switch (currentIndex) {
|
|
case 0:
|
|
AdaptivePageLayout.of(context)
|
|
.pushNamedAndRemoveUntilIsFirst('/newstatus');
|
|
break;
|
|
case 1:
|
|
AdaptivePageLayout.of(context)
|
|
.pushNamedAndRemoveUntilIsFirst('/newprivatechat');
|
|
break;
|
|
case 2:
|
|
AdaptivePageLayout.of(context)
|
|
.pushNamedAndRemoveUntilIsFirst('/newgroup');
|
|
break;
|
|
case 3:
|
|
_setServer(context);
|
|
break;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
_onShareContentChanged ??=
|
|
Matrix.of(context).onShareContentChanged.stream.listen(_onShare);
|
|
IconData fabIcon;
|
|
switch (currentIndex) {
|
|
case 0:
|
|
fabIcon = Icons.edit_outlined;
|
|
break;
|
|
case 1:
|
|
fabIcon = Icons.add_outlined;
|
|
break;
|
|
case 2:
|
|
fabIcon = Icons.group_add_outlined;
|
|
break;
|
|
case 3:
|
|
fabIcon = Icons.domain_outlined;
|
|
break;
|
|
}
|
|
|
|
return Scaffold(
|
|
appBar: appBar ??
|
|
AppBar(
|
|
centerTitle: false,
|
|
actions: [
|
|
IconButton(
|
|
icon: Icon(Icons.account_circle_outlined),
|
|
onPressed: () =>
|
|
AdaptivePageLayout.of(context).pushNamed('/settings'),
|
|
),
|
|
],
|
|
title: Text(AppConfig.applicationName)),
|
|
body: PreloadPageView(
|
|
controller: _pageController,
|
|
onPageChanged: (i) => setState(() => currentIndex = i),
|
|
children: [
|
|
StatusList(key: Key('StatusList')),
|
|
ChatList(
|
|
type: ChatListType.messages,
|
|
onCustomAppBar: (appBar) => setState(() => this.appBar = appBar),
|
|
),
|
|
ChatList(
|
|
type: ChatListType.groups,
|
|
onCustomAppBar: (appBar) => setState(() => this.appBar = appBar),
|
|
),
|
|
Discover(server: _server),
|
|
],
|
|
),
|
|
floatingActionButton: FloatingActionButton(
|
|
child: Icon(fabIcon),
|
|
onPressed: _onFabTab,
|
|
),
|
|
floatingActionButtonLocation: FloatingActionButtonLocation.centerDocked,
|
|
bottomNavigationBar: BottomNavigationBar(
|
|
unselectedItemColor: Theme.of(context).textTheme.bodyText1.color,
|
|
currentIndex: currentIndex,
|
|
showSelectedLabels: true,
|
|
showUnselectedLabels: false,
|
|
type: BottomNavigationBarType.fixed,
|
|
elevation: 20,
|
|
backgroundColor: Theme.of(context).appBarTheme.color,
|
|
onTap: (i) {
|
|
_pageController.animateToPage(
|
|
i,
|
|
duration: Duration(milliseconds: 200),
|
|
curve: Curves.bounceOut,
|
|
);
|
|
setState(() => currentIndex = i);
|
|
},
|
|
items: [
|
|
BottomNavigationBarItem(
|
|
label: L10n.of(context).status,
|
|
icon: Icon(Icons.home_outlined),
|
|
),
|
|
BottomNavigationBarItem(
|
|
label: L10n.of(context).messages,
|
|
icon: Icon(CupertinoIcons.chat_bubble_2),
|
|
),
|
|
BottomNavigationBarItem(
|
|
label: L10n.of(context).groups,
|
|
icon: Icon(Icons.people_outline),
|
|
),
|
|
BottomNavigationBarItem(
|
|
label: L10n.of(context).discover,
|
|
icon: Icon(CupertinoIcons.search_circle),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|