mirror of
https://github.com/Xevion/paperless-mobile.git
synced 2025-12-06 15:15:50 -06:00
374 lines
13 KiB
Dart
374 lines
13 KiB
Dart
import 'package:flutter/gestures.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_animate/flutter_animate.dart';
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
import 'package:flutter_svg/flutter_svg.dart';
|
|
import 'package:paperless_mobile/constants.dart';
|
|
import 'package:paperless_mobile/core/database/tables/local_user_account.dart';
|
|
import 'package:paperless_mobile/core/global/asset_images.dart';
|
|
import 'package:paperless_mobile/core/widgets/hint_card.dart';
|
|
import 'package:paperless_mobile/core/widgets/paperless_logo.dart';
|
|
import 'package:paperless_mobile/extensions/flutter_extensions.dart';
|
|
import 'package:paperless_mobile/features/documents/cubit/documents_cubit.dart';
|
|
import 'package:paperless_mobile/features/saved_view/cubit/saved_view_cubit.dart';
|
|
import 'package:paperless_mobile/features/sharing/cubit/receive_share_cubit.dart';
|
|
import 'package:paperless_mobile/generated/assets.gen.dart';
|
|
import 'package:paperless_mobile/generated/l10n/app_localizations.dart';
|
|
import 'package:paperless_mobile/routes/typed/branches/documents_route.dart';
|
|
import 'package:paperless_mobile/routes/typed/branches/saved_views_route.dart';
|
|
import 'package:paperless_mobile/routes/typed/branches/upload_queue_route.dart';
|
|
import 'package:paperless_mobile/routes/typed/shells/authenticated_route.dart';
|
|
import 'package:paperless_mobile/routes/typed/top_level/changelog_route.dart';
|
|
import 'package:paperless_mobile/routes/typed/top_level/settings_route.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'package:url_launcher/url_launcher_string.dart';
|
|
|
|
class AppDrawer extends StatelessWidget {
|
|
const AppDrawer({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final currentAccount = context.watch<LocalUserAccount>();
|
|
final username = currentAccount.paperlessUser.username;
|
|
final serverUrl =
|
|
currentAccount.serverUrl.replaceAll(RegExp(r'https?://'), '');
|
|
return SafeArea(
|
|
child: Drawer(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Row(
|
|
children: [
|
|
const PaperlessLogo.green(
|
|
width: 32,
|
|
height: 32,
|
|
),
|
|
Text(
|
|
"Paperless Mobile",
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
),
|
|
],
|
|
).paddedLTRB(8, 8, 8, 16),
|
|
Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
S.of(context)!.loggedInAs(username),
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
style: Theme.of(context).textTheme.labelMedium?.copyWith(
|
|
color: Theme.of(context)
|
|
.colorScheme
|
|
.onBackground
|
|
.withOpacity(0.5),
|
|
),
|
|
),
|
|
Text(
|
|
serverUrl,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
style: Theme.of(context).textTheme.labelMedium?.copyWith(
|
|
color: Theme.of(context)
|
|
.colorScheme
|
|
.onBackground
|
|
.withOpacity(0.5),
|
|
),
|
|
),
|
|
],
|
|
).paddedSymmetrically(horizontal: 16),
|
|
const Divider(),
|
|
ListTile(
|
|
dense: true,
|
|
title: Text(S.of(context)!.aboutThisApp),
|
|
leading: const Icon(Icons.info_outline),
|
|
onTap: () => _showAboutDialog(context),
|
|
),
|
|
ListTile(
|
|
dense: true,
|
|
leading: const Icon(Icons.favorite_outline),
|
|
title: Text(S.of(context)!.donate),
|
|
onTap: () {
|
|
showDialog(
|
|
context: context,
|
|
builder: (context) => AlertDialog(
|
|
icon: const Icon(Icons.favorite),
|
|
title: Text(S.of(context)!.donate),
|
|
content: Text(
|
|
S.of(context)!.donationDialogContent,
|
|
),
|
|
actionsAlignment: MainAxisAlignment.spaceBetween,
|
|
actions: [
|
|
const Text("~ Anton"),
|
|
TextButton(
|
|
onPressed: Navigator.of(context).pop,
|
|
child: Text(S.of(context)!.gotIt),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
},
|
|
),
|
|
ListTile(
|
|
dense: true,
|
|
leading: const Icon(Icons.history),
|
|
title: Text(S.of(context)!.changelog),
|
|
onTap: () {
|
|
ChangelogRoute().push(context);
|
|
},
|
|
),
|
|
ListTile(
|
|
dense: true,
|
|
leading: const Icon(Icons.bug_report_outlined),
|
|
title: Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
children: [
|
|
Text(S.of(context)!.reportABug),
|
|
const Icon(
|
|
Icons.open_in_new,
|
|
size: 16,
|
|
)
|
|
],
|
|
),
|
|
onTap: () {
|
|
launchUrlString(
|
|
'https://github.com/astubenbord/paperless-mobile/issues/new',
|
|
mode: LaunchMode.externalApplication,
|
|
);
|
|
},
|
|
),
|
|
ListTile(
|
|
dense: true,
|
|
leading: Assets.images.githubMark.svg(
|
|
colorFilter: ColorFilter.mode(
|
|
Theme.of(context).colorScheme.onBackground,
|
|
BlendMode.srcIn,
|
|
),
|
|
height: 24,
|
|
width: 24,
|
|
),
|
|
title: Text(S.of(context)!.sourceCode),
|
|
trailing: const Icon(
|
|
Icons.open_in_new,
|
|
size: 16,
|
|
),
|
|
onTap: () {
|
|
launchUrlString(
|
|
"https://github.com/astubenbord/paperless-mobile",
|
|
mode: LaunchMode.externalApplication,
|
|
);
|
|
},
|
|
),
|
|
Consumer<ConsumptionChangeNotifier>(
|
|
builder: (context, value, child) {
|
|
final files = value.pendingFiles;
|
|
final child = ListTile(
|
|
dense: true,
|
|
leading: const Icon(Icons.drive_folder_upload_outlined),
|
|
title: const Text("Pending Files"),
|
|
onTap: () {
|
|
UploadQueueRoute().push(context);
|
|
},
|
|
trailing: Text(
|
|
'${files.length}',
|
|
style: Theme.of(context).textTheme.bodyMedium,
|
|
),
|
|
);
|
|
if (files.isEmpty) {
|
|
return child;
|
|
}
|
|
return child
|
|
.animate(onPlay: (c) => c.repeat(reverse: true))
|
|
.fade(duration: 1.seconds, begin: 1, end: 0.3);
|
|
},
|
|
),
|
|
ListTile(
|
|
dense: true,
|
|
leading: const Icon(Icons.settings_outlined),
|
|
title: Text(
|
|
S.of(context)!.settings,
|
|
),
|
|
onTap: () => SettingsRoute().push(context),
|
|
),
|
|
const Divider(),
|
|
Text(
|
|
S.of(context)!.views,
|
|
textAlign: TextAlign.left,
|
|
style: Theme.of(context).textTheme.labelLarge,
|
|
).padded(16),
|
|
_buildSavedViews(),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildSavedViews() {
|
|
return BlocBuilder<SavedViewCubit, SavedViewState>(
|
|
builder: (context, state) {
|
|
return state.when(
|
|
initial: () => const SizedBox.shrink(),
|
|
loading: () => const Center(child: CircularProgressIndicator()),
|
|
loaded: (savedViews) {
|
|
final sidebarViews = savedViews.values
|
|
.where((element) => element.showInSidebar)
|
|
.toList();
|
|
if (sidebarViews.isEmpty) {
|
|
return Column(
|
|
children: [
|
|
Text(
|
|
S.of(context)!.youDidNotSaveAnyViewsYet,
|
|
style: Theme.of(context).textTheme.bodySmall,
|
|
).paddedOnly(
|
|
left: 16,
|
|
right: 16,
|
|
),
|
|
TextButton.icon(
|
|
onPressed: () {
|
|
Scaffold.of(context).closeDrawer();
|
|
const CreateSavedViewRoute(showInSidebar: true)
|
|
.push(context);
|
|
},
|
|
icon: const Icon(Icons.add),
|
|
label: Text(S.of(context)!.newView),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
return Expanded(
|
|
child: ListView.builder(
|
|
itemBuilder: (context, index) {
|
|
final view = sidebarViews[index];
|
|
return ListTile(
|
|
title: Text(view.name),
|
|
trailing: const Icon(Icons.arrow_forward),
|
|
onTap: () {
|
|
Scaffold.of(context).closeDrawer();
|
|
context
|
|
.read<DocumentsCubit>()
|
|
.updateFilter(filter: view.toDocumentFilter());
|
|
DocumentsRoute().go(context);
|
|
},
|
|
);
|
|
},
|
|
itemCount: sidebarViews.length,
|
|
),
|
|
);
|
|
},
|
|
error: () => Text(S.of(context)!.couldNotLoadSavedViews),
|
|
);
|
|
});
|
|
}
|
|
|
|
void _showAboutDialog(BuildContext context) {
|
|
final theme = Theme.of(context);
|
|
final colorScheme = theme.colorScheme;
|
|
showAboutDialog(
|
|
context: context,
|
|
applicationIcon: const ImageIcon(
|
|
AssetImage('assets/logos/paperless_logo_green.png'),
|
|
),
|
|
applicationName: 'Paperless Mobile',
|
|
applicationVersion: packageInfo.version + '+' + packageInfo.buildNumber,
|
|
children: [
|
|
Text(S.of(context)!.developedBy('Anton Stubenbord')),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
"Source Code",
|
|
style: theme.textTheme.titleMedium,
|
|
),
|
|
RichText(
|
|
text: TextSpan(
|
|
style: theme.textTheme.bodyMedium
|
|
?.copyWith(color: colorScheme.onSurface),
|
|
children: [
|
|
TextSpan(
|
|
text: S.of(context)!.findTheSourceCodeOn,
|
|
),
|
|
TextSpan(
|
|
text: ' GitHub',
|
|
style: const TextStyle(color: Colors.blue),
|
|
recognizer: TapGestureRecognizer()
|
|
..onTap = () {
|
|
launchUrlString(
|
|
'https://github.com/astubenbord/paperless-mobile',
|
|
mode: LaunchMode.externalApplication,
|
|
);
|
|
},
|
|
),
|
|
const TextSpan(text: '.'),
|
|
],
|
|
),
|
|
),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
'Credits',
|
|
style: theme.textTheme.titleMedium
|
|
?.copyWith(color: colorScheme.onSurface),
|
|
),
|
|
RichText(
|
|
text: TextSpan(
|
|
style: theme.textTheme.bodyMedium
|
|
?.copyWith(color: colorScheme.onSurface),
|
|
children: [
|
|
const TextSpan(
|
|
text: 'Onboarding images by ',
|
|
),
|
|
TextSpan(
|
|
text: 'pch.vector',
|
|
style: const TextStyle(color: Colors.blue),
|
|
recognizer: TapGestureRecognizer()
|
|
..onTap = () {
|
|
launchUrlString(
|
|
'https://www.freepik.com/free-vector/business-team-working-cogwheel-mechanism-together_8270974.htm#query=setting&position=4&from_view=author');
|
|
},
|
|
),
|
|
const TextSpan(
|
|
text: ' on Freepik.',
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
Widget _buildOnboardingImageCredits() {
|
|
return RichText(
|
|
text: TextSpan(
|
|
children: [
|
|
const TextSpan(
|
|
text: 'Onboarding images by ',
|
|
),
|
|
TextSpan(
|
|
text: 'pch.vector',
|
|
style: const TextStyle(color: Colors.blue),
|
|
recognizer: TapGestureRecognizer()
|
|
..onTap = () {
|
|
launchUrlString(
|
|
'https://www.freepik.com/free-vector/business-team-working-cogwheel-mechanism-together_8270974.htm#query=setting&position=4&from_view=author');
|
|
},
|
|
),
|
|
const TextSpan(
|
|
text: ' on Freepik.',
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
//Wrap(
|
|
// children: [
|
|
// const Text('Onboarding images by '),
|
|
// GestureDetector(
|
|
// onTap: followLink,
|
|
// child: RichText(
|
|
|
|
// 'pch.vector',
|
|
// style: TextStyle(color: Colors.blue),
|
|
// ),
|
|
// ),
|
|
// const Text(' on Freepik.')
|
|
// ],
|
|
// )
|