From 19682678dcebfd70ddf7825653415e74151d87a4 Mon Sep 17 00:00:00 2001 From: Jonas Sander <29028262+Jonas-Sander@users.noreply.github.com> Date: Sun, 21 Apr 2024 18:54:58 +0200 Subject: [PATCH] Update v2 announcement dialog (#1481) [v2 dialog.webm](https://github.com/SharezoneApp/sharezone-app/assets/29028262/b69d72c5-a2c1-4097-bd40-37b826d68845) Eltern/Lehrer: | ![Screenshot_1713715765](https://github.com/SharezoneApp/sharezone-app/assets/29028262/001702d5-fb76-4709-b12d-3dd1c18f2fc3) | ![Screenshot_1713715767](https://github.com/SharezoneApp/sharezone-app/assets/29028262/bc10dec5-ba08-44d5-8e85-dbca254e4d57) | | --- | --- | --- app/lib/dashboard/dashboard_page.dart | 2 +- app/lib/main/sharezone_app.dart | 11 +- .../sz_v2_annoucement_dialog.dart | 308 ----------- .../sz_v2_announcement_dialog.dart | 511 ++++++++++++++++++ lib/user/lib/src/models/user.dart | 41 +- 5 files changed, 543 insertions(+), 330 deletions(-) delete mode 100644 app/lib/sharezone_v2/sz_v2_annoucement_dialog.dart create mode 100644 app/lib/sharezone_v2/sz_v2_announcement_dialog.dart diff --git a/app/lib/dashboard/dashboard_page.dart b/app/lib/dashboard/dashboard_page.dart index 1771d73e2..2241951f5 100644 --- a/app/lib/dashboard/dashboard_page.dart +++ b/app/lib/dashboard/dashboard_page.dart @@ -41,7 +41,7 @@ import 'package:sharezone/navigation/scaffold/sharezone_custom_scaffold.dart'; import 'package:sharezone/overview/cache/profile_page_hint_cache.dart'; import 'package:sharezone/settings/src/subpages/changelog_page.dart'; import 'package:sharezone/settings/src/subpages/my_profile/change_state.dart'; -import 'package:sharezone/sharezone_v2/sz_v2_annoucement_dialog.dart'; +import 'package:sharezone/sharezone_v2/sz_v2_announcement_dialog.dart'; import 'package:sharezone/timetable/src/widgets/events/calender_event_card.dart'; import 'package:sharezone/timetable/src/widgets/events/event_view.dart'; import 'package:sharezone/timetable/timetable_add_event/timetable_add_event_dialog.dart'; diff --git a/app/lib/main/sharezone_app.dart b/app/lib/main/sharezone_app.dart index a9d38f38c..bd518f8a4 100644 --- a/app/lib/main/sharezone_app.dart +++ b/app/lib/main/sharezone_app.dart @@ -50,6 +50,7 @@ import 'package:sharezone/settings/src/subpages/notification.dart'; import 'package:sharezone/settings/src/subpages/about/about_page.dart'; import 'package:sharezone/settings/src/subpages/imprint/page/imprint_page.dart'; import 'package:sharezone/settings/src/subpages/theme/theme_page.dart'; +import 'package:sharezone/sharezone_v2/sz_v2_announcement_dialog.dart'; import 'package:sharezone/support/support_page.dart'; import 'package:sharezone/settings/src/subpages/timetable/timetable_settings_page.dart'; import 'package:sharezone/settings/src/subpages/web_app.dart'; @@ -149,10 +150,12 @@ class _SharezoneAppState extends State blocDependencies: widget.blocDependencies, home: MissingAccountInformationGuard( userCollection: widget.blocDependencies.references.users, - child: OnboardingListener( - child: NavigationController( - fbMessagingConfigurator: fbMessagingConfigurator, - key: navigationBloc.controllerKey, + child: SharezoneV2AnnoucementDialogGuard( + child: OnboardingListener( + child: NavigationController( + fbMessagingConfigurator: fbMessagingConfigurator, + key: navigationBloc.controllerKey, + ), ), ), ), diff --git a/app/lib/sharezone_v2/sz_v2_annoucement_dialog.dart b/app/lib/sharezone_v2/sz_v2_annoucement_dialog.dart deleted file mode 100644 index d6180ec41..000000000 --- a/app/lib/sharezone_v2/sz_v2_annoucement_dialog.dart +++ /dev/null @@ -1,308 +0,0 @@ -// Copyright (c) 2024 Sharezone UG (haftungsbeschränkt) -// Licensed under the EUPL-1.2-or-later. -// -// You may obtain a copy of the Licence at: -// https://joinup.ec.europa.eu/software/page/eupl -// -// SPDX-License-Identifier: EUPL-1.2 - -import 'package:bloc_provider/bloc_provider.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_markdown/flutter_markdown.dart'; -import 'package:sharezone/main/application_bloc.dart'; -import 'package:sharezone/navigation/logic/navigation_bloc.dart'; -import 'package:sharezone/navigation/models/navigation_item.dart'; -import 'package:sharezone_widgets/sharezone_widgets.dart'; - -Future openSzV2AnnoucementDialog(BuildContext context) async { - await Navigator.push( - context, MaterialPageRoute(builder: (context) => const _Dialog())); -} - -class _Dialog extends StatefulWidget { - const _Dialog(); - - @override - State<_Dialog> createState() => _DialogState(); -} - -class _DialogState extends State<_Dialog> { - bool _allCheckboxesChecked = false; - late final PageController controller; - - @override - void initState() { - super.initState(); - controller = PageController(); - } - - @override - Widget build(BuildContext context) { - return PopScope( - canPop: false, - child: Scaffold( - appBar: AppBar( - centerTitle: true, - title: const Text('Sharezone v2.0'), - elevation: 0, - automaticallyImplyLeading: false, - ), - body: MaxWidthConstraintBox( - maxWidth: 800, - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 20.0), - child: PageView( - controller: controller, - children: [ - const _JustText(markdownText: _markdownText1), - const _JustText(markdownText: _markdownText2), - const _JustText(markdownText: _markdownText3), - _FinalPage(onCheckboxesChanged: (allChecked) { - setState(() { - _allCheckboxesChecked = allChecked; - }); - }), - ], - ), - ), - ), - bottomNavigationBar: ListenableBuilder( - listenable: controller, - builder: (context, _) { - const lastPage = 3; - final bool isLastPage = controller.page == lastPage; - - return Padding( - padding: const EdgeInsets.only(bottom: 10), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceAround, - children: [ - Visibility( - maintainSize: true, - maintainAnimation: true, - maintainState: true, - visible: (controller.page ?? 0) > 0, - child: TextButton( - child: const Text('Zurück'), - onPressed: () { - controller.previousPage( - duration: const Duration(milliseconds: 250), - curve: Curves.easeInOut); - }, - ), - ), - ElevatedButton( - onPressed: !isLastPage || _allCheckboxesChecked - ? () async { - if (controller.page == lastPage) { - final ctx = BlocProvider.of( - context); - // ignore: unused_local_variable - final uid = ctx.api.uID; - - try { - // await FirebaseFirestore.instance - // .collection('users') - // .doc(uid) - // .update({ - // 'legal': { - // 'v2.0-terms-accepted': { - // 'deviceTime': clock.now(), - // 'serverTime': - // FieldValue.serverTimestamp(), - // }, - // }, - // }); - // ignore: use_build_context_synchronously - Navigator.pop(context); - // ignore: use_build_context_synchronously - BlocProvider.of(context) - .navigateTo( - NavigationItem.sharezonePlus); - } on Exception catch (e) { - // ignore: use_build_context_synchronously - await showLeftRightAdaptiveDialog( - context: context, - title: 'Fehler', - content: Text( - 'Es ist ein Fehler aufgetreten: $e. Falls dieser bestehen bleibt, dann schreibe uns unter support@sharezone.net')); - } - } else { - controller.nextPage( - duration: - const Duration(milliseconds: 250), - curve: Curves.easeInOut); - } - } - : null, - child: Text(isLastPage ? 'Fertig' : 'Weiter'), - ), - ], - ), - ); - })), - ); - } -} - -class _JustText extends StatelessWidget { - const _JustText({required this.markdownText}); - - final String markdownText; - - @override - Widget build(BuildContext context) { - return Align( - alignment: Alignment.centerLeft, - child: MarkdownBody(data: markdownText), - ); - } -} - -class _FinalPage extends StatefulWidget { - const _FinalPage({required this.onCheckboxesChanged}); - - final void Function(bool allCheckboxesChecked) onCheckboxesChanged; - - @override - State<_FinalPage> createState() => _FinalPageState(); -} - -class _FinalPageState extends State<_FinalPage> - with AutomaticKeepAliveClientMixin<_FinalPage> { - bool _box1Checked = false; - bool _box2Checked = false; - - // So that checkbox state is kept when going back from the last page - @override - bool get wantKeepAlive => true; - - void _onCheckboxChanged() { - widget.onCheckboxesChanged(_box1Checked && _box2Checked); - } - - @override - Widget build(BuildContext context) { - super.build(context); - return Column( - mainAxisAlignment: MainAxisAlignment.center, - children: [ - const _JustText(markdownText: _markdownText4), - const SizedBox( - height: 30, - ), - _Checkbox( - text: 'Ich habe die ANB gelesen und akzeptiere diese.', - value: _box1Checked, - onChanged: (newVal) { - setState(() { - _box1Checked = newVal; - }); - _onCheckboxChanged(); - }, - ), - _Checkbox( - text: - 'Ich habe zur Kenntnis genommen, dass die "Sharezone UG (haftungsbeschränkt)" Sharezone betreibt.', - value: _box2Checked, - onChanged: (newVal) { - setState(() { - _box2Checked = newVal; - }); - _onCheckboxChanged(); - }, - ), - const SizedBox( - height: 30, - ), - const _JustText( - markdownText: 'Deine personenbezogenen Daten werden gemäß unserer ' - '[Datenschutzerklärung](https://sharezone.net/datenschutz) verarbeitet.', - ) - ], - ); - } -} - -class _Checkbox extends StatelessWidget { - const _Checkbox({ - required this.text, - required this.value, - required this.onChanged, - }); - - final String text; - final bool value; - final void Function(bool) onChanged; - - @override - Widget build(BuildContext context) { - return SizedBox( - width: 600, - child: ListTile( - contentPadding: EdgeInsets.zero, - onTap: () => onChanged(!value), - leading: Checkbox( - value: value, - onChanged: (newVal) => onChanged(newVal!), - ), - title: MarkdownBody(data: text), - ), - ); - } -} - -const _markdownText1 = ''' -Hey du, schön dich hier zu haben! :) - -Wir haben Sharezone bis jetzt mit Herz, Blut und Tränen kostenlos für euch betrieben, weil wir vor allem anderen erstmal eine geile Schulapp machen wollten. -Wir freuen uns sehr, dass es so gut bei euch ankommt und ihr es so fleißig nutzt 💙🫶 - -Jetzt ist der Zeitpunkt gekommen, dass Sharezone sich selbst finanziert und es dadurch langfristig weiterlaufen kann 🏁🏃 - -Wie genau, das verraten wir dir, wenn du auf "Weiter" klickst ;) - -'''; - -const _markdownText2 = ''' -**Sharezone Plus** -Sharezone Plus ist ein Abbonement, mit welchem du “Plus-Features” freischalten kannst. - -Zum Beispiel hast du dadurch mehr Speicherplatz in der Dateiablage oder kannst unkomprimiert Bilder hochladen. - -Du kannst auch ohne Sharezone Plus weiterhin Sharezone kostenlos nutzen, allerdings mit ein paar kleinen Einschränkungen. - -Das Abo kann ganz einfach online von z.B. deinen Eltern per Bezahl-Link bezahlt werden. -'''; - -const _markdownText3 = ''' -Außerdem gibt es folgende Änderungen: - -**Version 2.0** -Wir haben das Design für dich überarbeitet, eine neue Navigation eingeführt und ein paar kleine Verbesserungen eingebaut. -Lass uns doch Feedback da, wie es dir gefällt. - -**Geänderte Rechtsform** -Sharezone läuft nun nicht mehr unter der "Sander, Jonas; Reichardt, Nils; Weuthen, Felix „Sharezone“ GbR", sondern unter der “Sharezone UG (haftungsbeschränkt)”. - -**Überarbeitung der Datenschutzerklärung** -Wir haben die Datenschutzerklärung einmal ganz neu überarbeitet und detailliert beschrieben, wie deine Daten verarbeitet und geschützt werden. -Für Sharezone Plus mussten wir außerdem neue externe Dienste einbinden (z.B. für die Zahlungsabwicklung). - -**Allgemeine Nutzungsbedingungen** -Wir haben neue allgemeinen Nutzungsbedingungen (“ANB”), die für die zukünftige Nutzung von Sharezone akzeptiert werden müssen. - -Diese regeln z.B., dass keine gewaltverherrlichenden Inhalte hochgeladen werden dürfen. Wir hoffen, dass das auch vorher klar war 😅 -'''; -const _markdownText4 = ''' -**Das war's!** - -Damit du weitermachen kannst, brauchen wir noch deine Zustimmung zu den unten aufgeführten Punkten. - -Falls du keine Einstimmung geben willst, dann kannst du hier dein Konto löschen oder den Support kontaktieren. - -Wir danken dir, uns bis hierhin begleitet zu haben. - -Euer Sharezone-Team 💙 - -'''; diff --git a/app/lib/sharezone_v2/sz_v2_announcement_dialog.dart b/app/lib/sharezone_v2/sz_v2_announcement_dialog.dart new file mode 100644 index 000000000..2e7111fb0 --- /dev/null +++ b/app/lib/sharezone_v2/sz_v2_announcement_dialog.dart @@ -0,0 +1,511 @@ +// Copyright (c) 2024 Sharezone UG (haftungsbeschränkt) +// Licensed under the EUPL-1.2-or-later. +// +// You may obtain a copy of the Licence at: +// https://joinup.ec.europa.eu/software/page/eupl +// +// SPDX-License-Identifier: EUPL-1.2 + +import 'dart:developer'; + +import 'package:bloc_provider/bloc_provider.dart'; +import 'package:build_context/build_context.dart'; +import 'package:clock/clock.dart'; +import 'package:cloud_firestore/cloud_firestore.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_markdown/flutter_markdown.dart'; +import 'package:sharezone/dashboard/dashboard_page.dart'; +import 'package:sharezone/legal/privacy_policy/privacy_policy_page.dart'; +import 'package:sharezone/legal/terms_of_service/terms_of_service_page.dart'; +import 'package:sharezone/main/application_bloc.dart'; +import 'package:sharezone/main/sharezone.dart'; +import 'package:sharezone/navigation/logic/navigation_bloc.dart'; +import 'package:sharezone/navigation/models/navigation_item.dart'; +import 'package:sharezone/support/support_page.dart'; +import 'package:sharezone_plus_page_ui/sharezone_plus_page_ui.dart'; +import 'package:sharezone_widgets/sharezone_widgets.dart'; +import 'package:user/user.dart'; + +Future openSzV2AnnoucementDialog(BuildContext context) async { + await Navigator.push( + context, + MaterialPageRoute( + builder: (context) => _StudentDialog( + BlocProvider.of(context) + .api + .user + .data! + .typeOfUser + .isStudent, + ))); +} + +const _skipKey = 'v2-dialog-skipped-on'; + +class SharezoneV2AnnoucementDialogGuard extends StatelessWidget { + const SharezoneV2AnnoucementDialogGuard({ + super.key, + required this.child, + }); + + final Widget child; + + @override + Widget build(BuildContext context) { + final szContext = BlocProvider.of(context); + + bool skip = false; + try { + final dialogSkippedNum = szContext.sharedPreferences.getInt(_skipKey); + DateTime? dialogSkipped = dialogSkippedNum != null + ? DateTime.fromMillisecondsSinceEpoch(dialogSkippedNum) + : null; + skip = dialogSkipped != null && + dialogSkipped.isAfter(clock.now().subtract(const Duration(days: 3))); + } catch (e) { + log('Error reading dialog skipped time: $e'); + } + + return StreamBuilder( + stream: szContext.api.user.userStream, + builder: (context, snapshot) { + if (isIntegrationTest) { + return child; + } + if (snapshot.hasData) { + final user = snapshot.data; + if (user != null && + user.legalData['v2_0-legal-accepted'] == null && + !skip) { + return _StudentDialog(user.typeOfUser.isStudent); + } + } + return child; + }); + } +} + +class _StudentDialog extends StatefulWidget { + const _StudentDialog(this.isStudent); + + final bool isStudent; + + @override + State<_StudentDialog> createState() => _StudentDialogState(); +} + +class _StudentDialogState extends State<_StudentDialog> { + bool _allCheckboxesChecked = false; + late final PageController controller; + + @override + void initState() { + super.initState(); + controller = PageController(); + } + + @override + Widget build(BuildContext context) { + final szContext = BlocProvider.of(context); + return PopScope( + canPop: false, + child: Scaffold( + appBar: AppBar( + centerTitle: true, + title: const Text('Sharezone v2.0'), + elevation: 0, + automaticallyImplyLeading: false, + ), + body: MaxWidthConstraintBox( + maxWidth: 800, + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 20.0), + child: PageView( + controller: controller, + children: [ + if (widget.isStudent) ...[ + const _JustText(markdownText: _markdownText1), + const _SharezonePlus(), + ], + _OtherChanges(widget.isStudent), + _FinalPage( + isStudent: widget.isStudent, + onCheckboxesChanged: (allChecked) { + setState(() { + _allCheckboxesChecked = allChecked; + }); + }), + ], + ), + ), + ), + bottomNavigationBar: ListenableBuilder( + listenable: controller, + builder: (context, _) { + final lastPage = widget.isStudent ? 3 : 1; + final bool isLastPage = controller.page == lastPage; + + return Padding( + padding: const EdgeInsets.only(bottom: 10), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceAround, + children: [ + Visibility( + maintainSize: true, + maintainAnimation: true, + maintainState: true, + visible: (controller.page ?? 0) > 0, + child: TextButton( + child: Text( + 'Zurück', + style: TextStyle( + color: context.isDarkThemeEnabled + ? null + : primaryColor, + ), + ), + onPressed: () { + controller.previousPage( + duration: const Duration(milliseconds: 250), + curve: Curves.easeInOut); + }, + ), + ), + ElevatedButton( + onPressed: !isLastPage || _allCheckboxesChecked + ? () async { + if (controller.page == lastPage) { + final ctx = BlocProvider.of( + context); + // We navigate beforehand, so that the context + // is not already unmounted when we try to + // navigate. + BlocProvider.of(context) + .navigateTo(NavigationItem.sharezonePlus); + // ignore: unused_local_variable + final uid = ctx.api.uID; + + try { + szContext.api.references.firestore + .collection('User') + .doc(uid) + .update({ + 'legal': { + 'v2_0-legal-accepted': { + "source": "v2.0-legal-dialog", + 'deviceTime': clock.now(), + 'serverTime': + FieldValue.serverTimestamp(), + }, + }, + }); + } on Exception catch (e) { + if (context.mounted) { + await showLeftRightAdaptiveDialog( + context: context, + title: 'Fehler', + content: Text( + 'Es ist ein Fehler aufgetreten: $e. Falls dieser bestehen bleibt, dann schreibe uns unter support@sharezone.net')); + } + } + } else { + controller.nextPage( + duration: + const Duration(milliseconds: 250), + curve: Curves.easeInOut); + } + } + : null, + child: Text( + isLastPage ? 'Fertig' : 'Weiter', + style: TextStyle( + color: context.isDarkThemeEnabled + ? null + : Colors.white), + ), + ), + ], + ), + ); + })), + ); + } +} + +class _SharezonePlus extends StatelessWidget { + const _SharezonePlus(); + + @override + Widget build(BuildContext context) { + return const SingleChildScrollView( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + SizedBox(height: 190), + _JustText(markdownText: _markdownText2), + SizedBox(height: 30), + SharezonePlusAdvantages( + isHomeworkDoneListsFeatureVisible: false, + isHomeworkReminderFeatureVisible: false, + ) + ], + ), + ); + } +} + +class _OtherChanges extends StatelessWidget { + const _OtherChanges(this.isStudent); + + final bool isStudent; + + @override + Widget build(BuildContext context) { + return Center( + child: SingleChildScrollView( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + if (isStudent) + const _JustText(markdownText: '## Weitere Änderungen'), + if (!isStudent) ...[ + const _JustText(markdownText: '## Wichtige Änderungen'), + const SizedBox(height: 10), + const _JustText( + markdownText: ''' +Hallo, hier ist das Sharezone Team :) Wir haben ein paar wichtige Änderungen, die wir dir gerne mitteilen möchten. +''', + ), + ], + const SizedBox(height: 10), + const _Card( + header: Text('Geänderte Rechtsform'), + body: MarkdownBody( + data: + 'Sharezone läuft nun nicht mehr unter der "Sander, Jonas; Reichardt, Nils; Weuthen, Felix „Sharezone“ GbR", sondern unter der “Sharezone UG (haftungsbeschränkt)”.'), + ), + const SizedBox(height: 12), + _Card( + header: const Text('Überarbeitung der Datenschutzerklärung'), + body: MarkdownBody( + data: + 'Wir haben die [Datenschutzerklärung](privacy-policy) einmal ganz neu überarbeitet und detailliert beschrieben, wie deine Daten verarbeitet und geschützt werden.${isStudent ? ' Für Sharezone Plus mussten wir außerdem neue externe Dienste einbinden (z.B. für die Zahlungsabwicklung oder verschicken von Emails).' : ''}', + onTapLink: (text, href, title) { + Navigator.of(context).pushNamed(PrivacyPolicyPage.tag); + }, + ), + ), + const SizedBox(height: 12), + _Card( + header: const Text('Allgemeine Nutzungsbedingungen (ANB)'), + body: MarkdownBody( + data: + 'Wir haben neue [allgemeine Nutzungsbedingungen (“ANB”)](terms-of-service), die für die zukünftige Nutzung von Sharezone akzeptiert werden müssen.', + onTapLink: (text, href, title) { + Navigator.of(context).pushNamed(TermsOfServicePage.tag); + }, + ), + ), + ], + ), + ), + ); + } +} + +class _Card extends StatelessWidget { + const _Card({ + required this.header, + required this.body, + }); + + final Widget header; + final Widget body; + + @override + Widget build(BuildContext context) { + return ExpansionCard( + header: header, + body: body, + backgroundColor: context.isDarkThemeEnabled + ? Theme.of(context).colorScheme.primary.withOpacity(0.2) + : primaryColor.withOpacity(.3), + ); + } +} + +class _JustText extends StatelessWidget { + const _JustText({required this.markdownText, this.onLinkTap}); + + final String markdownText; + final MarkdownTapLinkCallback? onLinkTap; + + @override + Widget build(BuildContext context) { + return Align( + alignment: Alignment.centerLeft, + child: MarkdownBody( + data: markdownText, + onTapLink: onLinkTap, + ), + ); + } +} + +class _FinalPage extends StatefulWidget { + const _FinalPage( + {required this.onCheckboxesChanged, required this.isStudent}); + + final void Function(bool allCheckboxesChecked) onCheckboxesChanged; + final bool isStudent; + + @override + State<_FinalPage> createState() => _FinalPageState(); +} + +class _FinalPageState extends State<_FinalPage> + with AutomaticKeepAliveClientMixin<_FinalPage> { + bool _box1Checked = false; + bool _box2Checked = false; + + // So that checkbox state is kept when going back from the last page + @override + bool get wantKeepAlive => true; + + void _onCheckboxChanged() { + widget.onCheckboxesChanged(_box1Checked && _box2Checked); + } + + @override + Widget build(BuildContext context) { + super.build(context); + return Center( + child: SingleChildScrollView( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + _JustText( + markdownText: ''' +**Das war's!** + +Damit du weitermachen kannst, brauchen wir noch deine Zustimmung zu den unten aufgeführten Punkten. + +Falls du keine Einstimmung geben willst, dann kannst du [hier](other-options) den Support kontaktieren. +${widget.isStudent ? '\n\nWir danken dir, uns bis hierhin begleitet zu haben.' : ''} + + Euer Sharezone-Team 💙''', + onLinkTap: (text, href, title) { + if (href == 'other-options') { + // ignore: use_build_context_synchronously + Navigator.of(context).pushNamed(SupportPage.tag); + } + }, + ), + const SizedBox(height: 30), + _Checkbox( + text: 'Ich habe [die ANB](anb) gelesen und akzeptiere diese.', + value: _box1Checked, + onChanged: (newVal) { + setState(() { + _box1Checked = newVal; + }); + _onCheckboxChanged(); + }, + onLinkTap: (text, href, title) { + if (href == 'anb') { + Navigator.of(context).pushNamed(TermsOfServicePage.tag); + } + }, + ), + _Checkbox( + text: + 'Ich habe zur Kenntnis genommen, dass die "Sharezone UG (haftungsbeschränkt)" Sharezone betreibt.', + value: _box2Checked, + onChanged: (newVal) { + setState(() { + _box2Checked = newVal; + }); + _onCheckboxChanged(); + }, + ), + const SizedBox(height: 30), + _JustText( + markdownText: + 'Deine personenbezogenen Daten werden gemäß unserer aktualisierten ' + '[Datenschutzerklärung](https://sharezone.net/datenschutz) verarbeitet.', + onLinkTap: (text, href, title) { + Navigator.of(context).pushNamed(PrivacyPolicyPage.tag); + }, + ), + const SizedBox(height: 10), + if (clock.now().isBefore(DateTime(2024, 05, 15))) + TextButton( + onPressed: () { + BlocProvider.of(context) + .sharedPreferences + .setInt(_skipKey, clock.now().millisecondsSinceEpoch); + Navigator.of(context).popAndPushNamed(DashboardPage.tag); + }, + child: const Text('Überspringen'), + ), + ], + ), + ), + ); + } +} + +class _Checkbox extends StatelessWidget { + const _Checkbox({ + required this.text, + required this.value, + required this.onChanged, + this.onLinkTap, + }); + + final String text; + final bool value; + final void Function(bool) onChanged; + final MarkdownTapLinkCallback? onLinkTap; + + @override + Widget build(BuildContext context) { + return SizedBox( + width: 600, + child: ListTile( + contentPadding: EdgeInsets.zero, + onTap: () => onChanged(!value), + leading: Checkbox( + value: value, + onChanged: (newVal) => onChanged(newVal!), + ), + title: MarkdownBody(data: text, onTapLink: onLinkTap), + ), + ); + } +} + +const _markdownText1 = ''' +Hey du, schön dich hier zu haben! :) + +Wir haben Sharezone bis jetzt mit Herz, Blut und Tränen kostenlos für euch betrieben, weil wir vor allem anderen erstmal eine geile Schulapp machen wollten. + +Wir freuen uns sehr, dass es so gut bei euch ankommt und ihr es so fleißig nutzt 💙🫶 + +Jetzt ist der Zeitpunkt gekommen, dass Sharezone sich selbst finanziert und es dadurch langfristig weiterlaufen kann 🏁🏃 + +Wie genau, das verraten wir dir, wenn du auf "Weiter" klickst ;) + +'''; + +const _markdownText2 = ''' +## Sharezone Plus + +Sharezone Plus bietet dir die Möglichkeit “Plus-Features” zu erwerben. + +Damit kannst du zum Beispiel deine Noten verwalten, hast mehr Speicherplatz in der Dateiablage oder kannst unkomprimiert Bilder hochladen. + +Du kannst die App auch ohne Sharezone Plus weiterhin kostenlos nutzen, mit ein paar kleinen Einschränkungen. + +Per Bezahl-Link kannst du Sharezone Plus auch ganz einfach online von z.B. deinen Eltern kaufen lassen. +'''; diff --git a/lib/user/lib/src/models/user.dart b/lib/user/lib/src/models/user.dart index f8daf700d..f715ec186 100644 --- a/lib/user/lib/src/models/user.dart +++ b/lib/user/lib/src/models/user.dart @@ -38,6 +38,8 @@ class AppUser { final Features? features; final Subscription? subscription; + final Map legalData; + AppUser._({ required this.id, required this.name, @@ -56,27 +58,28 @@ class AppUser { required this.createdOn, this.features, required this.subscription, + required this.legalData, }); factory AppUser.create({required String id}) { return AppUser._( - id: id, - name: "Anonymer Account", - abbreviation: "AA", - typeOfUser: TypeOfUser.student, - notificationTokens: [], - reminderTime: "18:00", - referralLink: null, - referredBy: null, - referralScore: 0, - state: StateEnum.anonymous, - blackboardNotifications: true, - commentsNotifications: true, - userSettings: UserSettings.defaultSettings(), - userTipData: UserTipData.empty(), - createdOn: null, - subscription: null, - ); + id: id, + name: "Anonymer Account", + abbreviation: "AA", + typeOfUser: TypeOfUser.student, + notificationTokens: [], + reminderTime: "18:00", + referralLink: null, + referredBy: null, + referralScore: 0, + state: StateEnum.anonymous, + blackboardNotifications: true, + commentsNotifications: true, + userSettings: UserSettings.defaultSettings(), + userTipData: UserTipData.empty(), + createdOn: null, + subscription: null, + legalData: {}); } factory AppUser.fromData(Map? data, {required String id}) { @@ -98,6 +101,7 @@ class AppUser { userTipData: UserTipData.empty(), createdOn: null, subscription: null, + legalData: {}, ); } return AppUser._( @@ -119,6 +123,8 @@ class AppUser { createdOn: dateTimeFromTimestampOrNull(data['createdOn']), features: Features.fromJson(data['features']), subscription: Subscription.fromData(data['subscription']), + legalData: decodeMap(data['legal'] ?? {}, + (key, decodedMapValue) => decodedMapValue as Map), ); } @@ -194,6 +200,7 @@ class AppUser { referredBy: referredBy, features: features ?? this.features, subscription: subscription ?? this.subscription, + legalData: legalData, ); } }