kasy-cli 1.20.0 → 1.20.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. package/README.md +11 -3
  2. package/lib/commands/new.js +78 -37
  3. package/lib/commands/run.js +7 -0
  4. package/lib/scaffold/backends/api/patch/lib/core/data/api/meta_ads_api.dart +1 -1
  5. package/lib/scaffold/backends/api/patch/lib/core/data/api/storage_api.dart +1 -1
  6. package/lib/scaffold/backends/api/patch/lib/core/data/entities/user_entity.dart +1 -1
  7. package/lib/scaffold/backends/api/patch/lib/features/authentication/api/authentication_api.dart +4 -5
  8. package/lib/scaffold/backends/api/patch/lib/features/feedbacks/api/feature_request_api.dart +1 -1
  9. package/lib/scaffold/backends/api/patch/lib/features/feedbacks/api/feature_vote_api.dart +1 -1
  10. package/lib/scaffold/backends/api/patch/lib/features/llm_chat/api/llm_chat_api.dart +1 -1
  11. package/lib/scaffold/backends/api/patch/lib/features/llm_chat/providers/llm_chat_notifier.dart +317 -0
  12. package/lib/scaffold/backends/api/patch/lib/features/notifications/api/device_api.dart +40 -1
  13. package/lib/scaffold/backends/api/patch/lib/features/notifications/api/entities/notifications_entity.dart +2 -0
  14. package/lib/scaffold/backends/api/patch/lib/features/onboarding/api/entities/user_info_entity.dart +1 -1
  15. package/lib/scaffold/backends/api/patch/lib/features/subscription/api/entities/subscription_entity.dart +1 -1
  16. package/lib/scaffold/backends/api/patch/lib/features/subscription/shared/maybeshow_premium.dart +0 -2
  17. package/lib/scaffold/backends/api/pubspec.yaml.tpl +2 -0
  18. package/lib/scaffold/backends/firebase/enable-auth-via-cli.js +11 -8
  19. package/lib/scaffold/backends/supabase/deploy.js +56 -3
  20. package/lib/scaffold/backends/supabase/patch/lib/core/data/api/storage_api.dart +5 -11
  21. package/lib/scaffold/backends/supabase/patch/lib/core/data/entities/user_entity.dart +2 -2
  22. package/lib/scaffold/backends/supabase/patch/lib/features/notifications/api/device_api.dart +31 -1
  23. package/lib/scaffold/backends/supabase/patch/lib/features/onboarding/api/entities/user_info_entity.dart +1 -1
  24. package/lib/scaffold/backends/supabase/patch/lib/features/subscription/api/entities/subscription_entity.dart +1 -1
  25. package/lib/scaffold/backends/supabase/pubspec.yaml.tpl +2 -0
  26. package/lib/scaffold/catalog.js +2 -2
  27. package/lib/scaffold/generate.js +19 -3
  28. package/lib/scaffold/shared/generator-utils.js +265 -55
  29. package/lib/scaffold/shared/post-build.js +11 -0
  30. package/lib/utils/i18n/messages-en.js +5 -1
  31. package/lib/utils/i18n/messages-es.js +5 -1
  32. package/lib/utils/i18n/messages-pt.js +5 -1
  33. package/package.json +1 -1
  34. package/templates/firebase/lib/components/kasy_sidebar_pro.dart +3 -1
  35. package/templates/firebase/lib/core/bottom_menu/bottom_menu.dart +38 -128
  36. package/templates/firebase/lib/core/bottom_menu/bottom_router.dart +6 -125
  37. package/templates/firebase/lib/core/states/user_state_notifier.dart +8 -10
  38. package/templates/firebase/lib/features/home/home_components_page.dart +8 -14
  39. package/templates/firebase/lib/features/home/home_page.dart +7 -8
  40. package/templates/firebase/lib/router.dart +60 -0
  41. package/templates/firebase/test/core/bottom_menu/detail_route_menu_test.dart +57 -0
  42. package/lib/scaffold/backends/api/patch/lib/core/rating/widgets/review_popup.dart +0 -211
  43. package/lib/scaffold/backends/api/patch/lib/features/notifications/providers/models/notification.dart +0 -185
  44. package/lib/scaffold/backends/api/patch/lib/features/onboarding/ui/components/onboarding_notifications_setup.dart +0 -73
  45. package/lib/scaffold/backends/api/patch/lib/main.dart +0 -275
  46. package/lib/scaffold/backends/api/patch/lib/router.dart +0 -133
  47. package/lib/scaffold/backends/supabase/patch/lib/core/rating/widgets/review_popup.dart +0 -211
  48. package/lib/scaffold/backends/supabase/patch/lib/features/feedbacks/ui/component/add_feature_form.dart +0 -199
  49. package/lib/scaffold/backends/supabase/patch/lib/features/notifications/providers/models/notification.dart +0 -174
  50. package/lib/scaffold/backends/supabase/patch/lib/features/onboarding/ui/components/onboarding_notifications_setup.dart +0 -73
  51. package/lib/scaffold/backends/supabase/patch/lib/main.dart +0 -307
  52. package/lib/scaffold/backends/supabase/patch/lib/router.dart +0 -133
@@ -1,211 +0,0 @@
1
- import 'package:flutter/material.dart';
2
- import 'package:flutter_animate/flutter_animate.dart';
3
- import 'package:flutter_riverpod/flutter_riverpod.dart';
4
- import 'package:go_router/go_router.dart';
5
- import 'package:kasy_kit/components/components.dart';
6
- import 'package:kasy_kit/core/data/api/analytics_api.dart';
7
- import 'package:kasy_kit/core/rating/providers/rating_repository.dart';
8
- import 'package:kasy_kit/core/states/user_state_notifier.dart';
9
- import 'package:kasy_kit/core/theme/theme.dart';
10
- import 'package:kasy_kit/core/widgets/responsive_layout.dart';
11
- import 'package:kasy_kit/i18n/translations.g.dart';
12
- import 'package:logger/logger.dart';
13
-
14
- Future<bool> showReviewDialog(WidgetRef ref, {bool force = false}) {
15
- if (!ref.context.mounted) {
16
- return Future.value(false);
17
- }
18
- final ratingRepository = ref.watch(ratingRepositoryProvider);
19
- final userState = ref.watch(userStateNotifierProvider);
20
- final ratingFuture = ratingRepository.getReview(userState.user);
21
-
22
- return ratingFuture.then((rating) {
23
- final shouldAsk = rating.shouldAsk();
24
- Logger().d('should Ask for review: $shouldAsk');
25
- if (!shouldAsk && !force) {
26
- return false;
27
- }
28
- if (!ref.context.mounted) {
29
- return false;
30
- }
31
- showDialog(
32
- context: ref.context,
33
- barrierDismissible: false,
34
- builder: (context) {
35
- ratingRepository.delay();
36
- return Animate(
37
- effects: const [
38
- FadeEffect(
39
- delay: Duration(milliseconds: 100),
40
- duration: Duration(milliseconds: 300),
41
- ),
42
- MoveEffect(
43
- delay: Duration(milliseconds: 100),
44
- duration: Duration(milliseconds: 450),
45
- curve: Curves.easeOut,
46
- begin: Offset(0, 50),
47
- end: Offset.zero,
48
- ),
49
- ],
50
- child: Dialog(
51
- backgroundColor: Colors.transparent,
52
- child: DeviceSizeBuilder(
53
- builder: (device) {
54
- final maxWidth = switch (device) {
55
- DeviceType.medium => MediaQuery.of(context).size.width - KasySpacing.xl,
56
- _ => 550.0,
57
- };
58
- return ConstrainedBox(
59
- constraints: BoxConstraints(maxWidth: maxWidth),
60
- child: Center(
61
- child: Container(
62
- decoration: BoxDecoration(
63
- color: context.colors.background,
64
- borderRadius: KasyRadius.lgBorderRadius,
65
- border: Border.all(
66
- color: context.colors.onBackground.withValues(
67
- alpha: .3,
68
- ),
69
- width: 2,
70
- strokeAlign: BorderSide.strokeAlignOutside,
71
- ),
72
- ),
73
- child: Column(
74
- mainAxisSize: MainAxisSize.min,
75
- crossAxisAlignment: CrossAxisAlignment.stretch,
76
- children: [
77
- Flexible(
78
- child: Stack(
79
- children: [
80
- ClipRRect(
81
- borderRadius: const BorderRadius.only(
82
- topLeft: Radius.circular(KasyRadius.lg),
83
- topRight: Radius.circular(KasyRadius.lg),
84
- ),
85
- child: Image.asset(
86
- 'assets/images/review.png',
87
- fit: BoxFit.fitWidth,
88
- width: maxWidth,
89
- ),
90
- ),
91
- Positioned(
92
- top: KasySpacing.md,
93
- left: KasySpacing.md,
94
- child: CloseIcon(
95
- onExit: () {
96
- ref
97
- .read(analyticsApiProvider)
98
- .logEvent('rating_popup_close', {});
99
- rating.delay().then((_) {
100
- if (!context.mounted) return;
101
- Navigator.of(context).pop();
102
- });
103
- },
104
- ),
105
- ),
106
- ],
107
- ),
108
- ),
109
- const SizedBox(height: KasySpacing.md),
110
- Flexible(
111
- flex: 0,
112
- child: Padding(
113
- padding: const EdgeInsets.symmetric(
114
- horizontal: KasySpacing.smd,
115
- ),
116
- child: Text(
117
- Translations.of(context).review_popup.title,
118
- textAlign: TextAlign.center,
119
- style: context.textTheme.headlineSmall,
120
- ),
121
- ),
122
- ),
123
- const SizedBox(height: KasySpacing.md),
124
- Padding(
125
- padding: const EdgeInsets.symmetric(
126
- horizontal: KasySpacing.lg,
127
- ),
128
- child: Text(
129
- Translations.of(context).review_popup.description,
130
- textAlign: TextAlign.center,
131
- style: context.textTheme.bodyMedium,
132
- ),
133
- ),
134
- const SizedBox(height: KasySpacing.lg),
135
- Padding(
136
- padding: const EdgeInsets.symmetric(
137
- horizontal: KasySpacing.md,
138
- ),
139
- child: KasyButton(
140
- label: Translations.of(context).review_popup.rate_button,
141
- expand: true,
142
- onPressed: () {
143
- ref.read(analyticsApiProvider).logEvent('rating_popup_show', {});
144
- ratingRepository
145
- .rate()
146
- .then((res) => rating.review())
147
- .then((_) {
148
- if (!context.mounted) return;
149
- Navigator.of(context).pop();
150
- });
151
- },
152
- ),
153
- ),
154
- const SizedBox(height: KasySpacing.sm),
155
- KasyButton(
156
- label: Translations.of(context).review_popup.cancel_button,
157
- variant: KasyButtonVariant.tertiary,
158
- expand: true,
159
- onPressed: () => Navigator.of(context).pop(true),
160
- ),
161
- const SizedBox(height: KasySpacing.sm),
162
- ],
163
- ),
164
- ),
165
- ),
166
- );
167
- },
168
- ),
169
- ),
170
- );
171
- },
172
- ).then((shouldOpenFeedbackPage) {
173
- if (shouldOpenFeedbackPage == true && ref.context.mounted) {
174
- ref.context.push("/feedback");
175
- }
176
- });
177
- return true;
178
- });
179
- }
180
-
181
- class CloseIcon extends StatelessWidget {
182
- final VoidCallback onExit;
183
-
184
- const CloseIcon({super.key, required this.onExit});
185
-
186
- @override
187
- Widget build(BuildContext context) {
188
- return ClipOval(
189
- child: Material(
190
- color: Colors.transparent,
191
- child: InkWell(
192
- onTap: () => onExit.call(),
193
- child: Ink(
194
- width: 32,
195
- height: 32,
196
- // padding: const EdgeInsets.all(6),
197
- decoration: BoxDecoration(
198
- color: context.colors.background,
199
- shape: BoxShape.circle,
200
- ),
201
- child: Icon(
202
- KasyIcons.close,
203
- color: context.colors.onBackground,
204
- size: 21,
205
- ),
206
- ),
207
- ),
208
- ),
209
- );
210
- }
211
- }
@@ -1,185 +0,0 @@
1
- // ignore_for_file: invalid_annotation_target, constant_identifier_names
2
-
3
- import 'package:kasy_kit/features/notifications/api/entities/notifications_entity.dart';
4
- import 'package:kasy_kit/features/notifications/api/local_notifier.dart';
5
- import 'package:kasy_kit/features/notifications/repositories/notifications_repository.dart';
6
- import 'package:go_router/go_router.dart';
7
- import 'package:kasy_kit/router.dart';
8
- import 'package:freezed_annotation/freezed_annotation.dart';
9
- import 'package:logger/logger.dart';
10
- import 'package:permission_handler/permission_handler.dart';
11
- import 'package:sentry_flutter/sentry_flutter.dart';
12
- import 'package:url_launcher/url_launcher.dart';
13
-
14
- part 'notification.freezed.dart';
15
- part 'notification.g.dart';
16
-
17
- @freezed
18
- sealed class Notification with _$Notification {
19
- const Notification._();
20
-
21
- const factory Notification.withData({
22
- String? id,
23
- required String title,
24
- required String body,
25
- required DateTime createdAt,
26
- DateTime? readAt,
27
- String? imageUrl,
28
- @JsonKey(includeFromJson: false, includeToJson: false)
29
- LocalNotifier? notifier,
30
- @JsonKey(includeFromJson: false, includeToJson: false)
31
- NotificationSettings? notifierSettings,
32
- NotificationTypes? type,
33
- Map<String, dynamic>? data,
34
- }) = NotificationData;
35
-
36
- factory Notification.fromJson(Map<String, dynamic> json) =>
37
- _$NotificationFromJson(json);
38
-
39
- factory Notification.from(
40
- Map<String, dynamic> json, {
41
- String? id,
42
- Map<String, dynamic>? data,
43
- LocalNotifier? notifierApi,
44
- NotificationSettings? notifierSettings,
45
- }) =>
46
- Notification.withData(
47
- id: id,
48
- title: json['title'] as String,
49
- body: json['body'] as String,
50
- imageUrl: json['image'] as String?,
51
- type: data != null && data.containsKey('type')
52
- ? NotificationTypes.values.firstWhere(
53
- (e) => e.name == data['type'],
54
- orElse: () => NotificationTypes.OTHER,
55
- )
56
- : null,
57
- data: data,
58
- createdAt: DateTime.now(),
59
- notifier: notifierApi,
60
- notifierSettings: notifierSettings,
61
- );
62
-
63
- Future<void> show({NotificationSettings? settings}) async {
64
- if (notifier == null) {
65
- throw Exception(
66
- 'You must provide a LocalNotifierApi to show a notification',
67
- );
68
- }
69
- if (notifierSettings != null) {
70
- await notifier!.show(notifierSettings!, this);
71
- return;
72
- } else if (settings != null) {
73
- await notifier!.show(settings, this);
74
- return;
75
- }
76
- throw Exception(
77
- 'You must provide a NotificationSettings to show a notification',
78
- );
79
- }
80
-
81
- bool get seen => readAt != null;
82
-
83
- Future<void> onTap() async {
84
- // if the app is not ready, we store the notification in the pending notification handler
85
- if (navigatorKey.currentContext == null) {
86
- return;
87
- }
88
-
89
- // Open external URL
90
- if (type == NotificationTypes.LINK && data?.containsKey('url') == true) {
91
- try {
92
- launchUrl(Uri.parse(data!['url'] as String));
93
- } catch (e, s) {
94
- Logger().e("error $e");
95
- Sentry.captureException(e, stackTrace: s);
96
- }
97
- return;
98
- }
99
- // Navigate to an internal route sent in the notification data.
100
- // Example: data: {"route": "/premium"} → opens the premium page.
101
- if (data?.containsKey('route') == true) {
102
- final route = data!['route'] as String;
103
- navigatorKey.currentContext!.go(route);
104
- return;
105
- }
106
- // Default: open the notifications list
107
- navigatorKey.currentContext!.go('/notifications');
108
- }
109
- }
110
-
111
- sealed class NotificationPermission {
112
- /// ask for permission if needed
113
- Future<void> maybeAsk() async {
114
- final permission = this;
115
- switch (permission) {
116
- case NotificationPermissionWaiting():
117
- await permission.ask();
118
- case NotificationPermissionDenied():
119
- await permission.ask();
120
- case NotificationPermissionGranted():
121
- default:
122
- }
123
- }
124
- }
125
-
126
- /// we asked for permission and it was granted
127
- class NotificationPermissionGranted extends NotificationPermission {
128
- NotificationPermissionGranted();
129
- }
130
-
131
- /// we asked for permission and it was denied
132
- class NotificationPermissionDenied extends NotificationPermission {
133
- final NotificationSettings? _notificationSettings;
134
- final NotificationsRepository? _repository;
135
-
136
- NotificationPermissionDenied({
137
- NotificationSettings? notificationSettings,
138
- NotificationsRepository? repository,
139
- }) : _repository = repository,
140
- _notificationSettings = notificationSettings;
141
-
142
- Future<void> ask() async {
143
- if (_notificationSettings == null) {
144
- throw Exception("NotificationsApi is null");
145
- }
146
- final granted = await _notificationSettings.askPermission();
147
- NotificationPermission? permission;
148
- if (granted) {
149
- permission = NotificationPermissionGranted();
150
- } else {
151
- permission = NotificationPermissionDenied();
152
- }
153
- await _repository!.init();
154
- }
155
-
156
- Future<void> openSettings() async {
157
- await openAppSettings();
158
- }
159
- }
160
-
161
- /// we never asked for permission
162
- class NotificationPermissionWaiting extends NotificationPermission {
163
- final NotificationSettings? _notificationSettings;
164
- final NotificationsRepository? _repository;
165
-
166
- NotificationPermissionWaiting({
167
- NotificationSettings? notificationSettings,
168
- NotificationsRepository? repository,
169
- }) : _notificationSettings = notificationSettings,
170
- _repository = repository;
171
-
172
- Future<void> ask() async {
173
- if (_notificationSettings == null) {
174
- throw Exception("NotificationsApi is null");
175
- }
176
- final granted = await _notificationSettings.askPermission();
177
- NotificationPermission? permission;
178
- if (granted) {
179
- permission = NotificationPermissionGranted();
180
- await _repository?.init();
181
- } else {
182
- permission = NotificationPermissionDenied();
183
- }
184
- }
185
- }
@@ -1,73 +0,0 @@
1
- import 'package:flutter/material.dart';
2
- import 'package:flutter_riverpod/flutter_riverpod.dart';
3
- import 'package:kasy_kit/components/components.dart';
4
- import 'package:kasy_kit/core/data/api/analytics_api.dart';
5
- import 'package:kasy_kit/core/theme/theme.dart';
6
- import 'package:kasy_kit/features/notifications/providers/models/notification.dart';
7
- import 'package:kasy_kit/features/notifications/repositories/notifications_repository.dart';
8
- import 'package:kasy_kit/features/onboarding/providers/onboarding_provider.dart';
9
- import 'package:kasy_kit/features/onboarding/ui/widgets/onboarding_background.dart';
10
- import 'package:kasy_kit/features/onboarding/ui/widgets/onboarding_illustration_scaffold.dart';
11
- import 'package:kasy_kit/i18n/translations.g.dart';
12
-
13
- final _formStep3Key = GlobalKey<FormState>();
14
-
15
- class NotificationsPermissionStep extends ConsumerWidget {
16
- final String nextRoute;
17
-
18
- const NotificationsPermissionStep({
19
- super.key,
20
- required this.nextRoute,
21
- });
22
-
23
- @override
24
- Widget build(BuildContext context, WidgetRef ref) {
25
- final notifRepository = ref.watch(notificationRepositoryProvider);
26
- final permissionFuture = notifRepository.getPermissionStatus();
27
- final translations = Translations.of(context).onboarding.notifications;
28
-
29
- return OnboardingBackground(
30
- child: FutureBuilder<NotificationPermission>(
31
- future: permissionFuture,
32
- builder: (context, snapshot) {
33
- if (!snapshot.hasData) {
34
- return const SizedBox();
35
- }
36
- return Form(
37
- key: _formStep3Key,
38
- child: OnboardingIllustrationScaffold(
39
- progress: 0.9,
40
- title: translations.title,
41
- description: translations.description,
42
- imageAsset: 'assets/images/onboarding/img2.jpg',
43
- footerActions: [
44
- KasyButton(
45
- label: translations.continue_button,
46
- expand: true,
47
- onPressed: () => ref.onboardingNotifier
48
- .setupNotifications()
49
- .then((_) {
50
- if (!context.mounted) return;
51
- Navigator.of(context).pushNamed(nextRoute);
52
- }),
53
- ),
54
- const SizedBox(height: KasySpacing.smd),
55
- KasyButton(
56
- label: translations.skip_button,
57
- variant: KasyButtonVariant.soft,
58
- expand: true,
59
- onPressed: () {
60
- ref
61
- .read(analyticsApiProvider)
62
- .logEvent('setup_notifications_refused', {});
63
- Navigator.of(context).pushNamed(nextRoute);
64
- },
65
- ),
66
- ],
67
- ),
68
- );
69
- },
70
- ),
71
- );
72
- }
73
- }