| 
									
										
										
										
											2023-08-14 13:34:01 -07:00
										 |  |  | import 'package:appflowy/generated/flowy_svgs.g.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  | import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart'; | 
					
						
							| 
									
										
										
										
											2023-11-29 19:01:29 -07:00
										 |  |  | import 'package:appflowy/mobile/presentation/database/card/card.dart'; | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | import 'package:appflowy/plugins/database/application/field/field_controller.dart'; | 
					
						
							| 
									
										
										
										
											2024-01-05 17:30:54 +08:00
										 |  |  | import 'package:appflowy/plugins/database/application/row/row_cache.dart'; | 
					
						
							| 
									
										
										
										
											2024-08-23 13:55:40 +08:00
										 |  |  | import 'package:appflowy/plugins/database/application/row/row_controller.dart'; | 
					
						
							| 
									
										
										
										
											2024-01-05 17:30:54 +08:00
										 |  |  | import 'package:appflowy/plugins/database/grid/presentation/widgets/row/action.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  | import 'package:appflowy/shared/af_image.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  | import 'package:appflowy/shared/flowy_gradient_colors.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  | import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart'; | 
					
						
							|  |  |  | import 'package:appflowy_backend/protobuf/flowy-user/user_profile.pb.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  | import 'package:appflowy_editor/appflowy_editor.dart'; | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | import 'package:collection/collection.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  | import 'package:flowy_infra/theme_extension.dart'; | 
					
						
							| 
									
										
										
										
											2022-09-19 15:36:59 +08:00
										 |  |  | import 'package:flowy_infra_ui/flowy_infra_ui.dart'; | 
					
						
							| 
									
										
										
										
											2023-05-31 16:52:37 +08:00
										 |  |  | import 'package:flowy_infra_ui/style_widget/hover.dart'; | 
					
						
							| 
									
										
										
										
											2025-03-04 11:29:38 +08:00
										 |  |  | import 'package:flutter/material.dart'; | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  | import 'package:flutter_bloc/flutter_bloc.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-12 14:40:19 +08:00
										 |  |  | import 'package:universal_platform/universal_platform.dart'; | 
					
						
							| 
									
										
										
										
											2023-11-26 15:10:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | import '../cell/card_cell_builder.dart'; | 
					
						
							|  |  |  | import '../cell/card_cell_skeleton/card_cell.dart'; | 
					
						
							| 
									
										
										
										
											2024-09-05 06:03:58 +02:00
										 |  |  | import 'card_bloc.dart'; | 
					
						
							| 
									
										
										
										
											2022-09-19 15:36:59 +08:00
										 |  |  | import 'container/accessory.dart'; | 
					
						
							|  |  |  | import 'container/card_container.dart'; | 
					
						
							| 
									
										
										
										
											2022-08-10 17:59:28 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | /// Edit a database row with card style widget
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | class RowCard extends StatefulWidget { | 
					
						
							| 
									
										
										
										
											2024-01-25 16:37:36 +01:00
										 |  |  |   const RowCard({ | 
					
						
							|  |  |  |     super.key, | 
					
						
							|  |  |  |     required this.fieldController, | 
					
						
							|  |  |  |     required this.rowMeta, | 
					
						
							|  |  |  |     required this.viewId, | 
					
						
							|  |  |  |     required this.isEditing, | 
					
						
							|  |  |  |     required this.rowCache, | 
					
						
							|  |  |  |     required this.cellBuilder, | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |     required this.onTap, | 
					
						
							| 
									
										
										
										
											2024-01-25 16:37:36 +01:00
										 |  |  |     required this.onStartEditing, | 
					
						
							|  |  |  |     required this.onEndEditing, | 
					
						
							|  |  |  |     required this.styleConfiguration, | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |     this.onShiftTap, | 
					
						
							| 
									
										
										
										
											2024-01-25 16:37:36 +01:00
										 |  |  |     this.groupingFieldId, | 
					
						
							|  |  |  |     this.groupId, | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  |     required this.userProfile, | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |     this.isCompact = false, | 
					
						
							| 
									
										
										
										
											2024-01-25 16:37:36 +01:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |   final FieldController fieldController; | 
					
						
							| 
									
										
										
										
											2023-06-14 22:16:33 +08:00
										 |  |  |   final RowMetaPB rowMeta; | 
					
						
							| 
									
										
										
										
											2023-01-31 08:28:31 +08:00
										 |  |  |   final String viewId; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  |   final String? groupingFieldId; | 
					
						
							| 
									
										
										
										
											2023-08-01 08:49:48 +05:30
										 |  |  |   final String? groupId; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-13 16:23:44 +08:00
										 |  |  |   final bool isEditing; | 
					
						
							| 
									
										
										
										
											2023-03-08 21:19:44 +08:00
										 |  |  |   final RowCache rowCache; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /// The [CardCellBuilder] is used to build the card cells.
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |   final CardCellBuilder cellBuilder; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /// Called when the user taps on the card.
 | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |   final void Function(BuildContext context) onTap; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   final void Function(BuildContext context)? onShiftTap; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /// Called when the user starts editing the card.
 | 
					
						
							| 
									
										
										
										
											2022-10-06 22:26:18 +08:00
										 |  |  |   final VoidCallback onStartEditing; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /// Called when the user ends editing the card.
 | 
					
						
							| 
									
										
										
										
											2022-10-06 22:26:18 +08:00
										 |  |  |   final VoidCallback onEndEditing; | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  |   final RowCardStyleConfiguration styleConfiguration; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |   /// Specifically the token is used to handle requests to retrieve images
 | 
					
						
							|  |  |  |   /// from cloud storage, such as the card cover.
 | 
					
						
							|  |  |  |   final UserProfilePB? userProfile; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /// Whether the card is in a narrow space.
 | 
					
						
							|  |  |  |   /// This is used to determine eg. the Cover height.
 | 
					
						
							|  |  |  |   final bool isCompact; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  |   @override | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |   State<RowCard> createState() => _RowCardState(); | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | class _RowCardState extends State<RowCard> { | 
					
						
							| 
									
										
										
										
											2023-11-24 00:25:12 +08:00
										 |  |  |   final popoverController = PopoverController(); | 
					
						
							| 
									
										
										
										
											2023-05-31 16:52:37 +08:00
										 |  |  |   late final CardBloc _cardBloc; | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   void initState() { | 
					
						
							| 
									
										
										
										
											2023-11-09 00:30:50 +01:00
										 |  |  |     super.initState(); | 
					
						
							| 
									
										
										
										
											2024-08-23 13:55:40 +08:00
										 |  |  |     final rowController = RowController( | 
					
						
							|  |  |  |       viewId: widget.viewId, | 
					
						
							|  |  |  |       rowMeta: widget.rowMeta, | 
					
						
							|  |  |  |       rowCache: widget.rowCache, | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-08 21:19:44 +08:00
										 |  |  |     _cardBloc = CardBloc( | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |       fieldController: widget.fieldController, | 
					
						
							| 
									
										
										
										
											2023-01-31 08:28:31 +08:00
										 |  |  |       viewId: widget.viewId, | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  |       groupFieldId: widget.groupingFieldId, | 
					
						
							| 
									
										
										
										
											2022-09-07 11:33:42 +08:00
										 |  |  |       isEditing: widget.isEditing, | 
					
						
							| 
									
										
										
										
											2024-08-23 13:55:40 +08:00
										 |  |  |       rowController: rowController, | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     )..add(const CardEvent.initial()); | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-09-07 11:33:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |   @override | 
					
						
							|  |  |  |   void didUpdateWidget(covariant oldWidget) { | 
					
						
							|  |  |  |     if (widget.isEditing != _cardBloc.state.isEditing) { | 
					
						
							|  |  |  |       _cardBloc.add(CardEvent.setIsEditing(widget.isEditing)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     super.didUpdateWidget(oldWidget); | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-08-10 17:59:28 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |   @override | 
					
						
							| 
									
										
										
										
											2024-01-29 10:26:45 +08:00
										 |  |  |   void dispose() { | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     _cardBloc.close(); | 
					
						
							|  |  |  |     super.dispose(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 17:59:28 +08:00
										 |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  |     return BlocProvider.value( | 
					
						
							|  |  |  |       value: _cardBloc, | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |       child: BlocListener<CardBloc, CardState>( | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |         listenWhen: (previous, current) => | 
					
						
							|  |  |  |             previous.isEditing != current.isEditing, | 
					
						
							|  |  |  |         listener: (context, state) { | 
					
						
							|  |  |  |           if (!state.isEditing) { | 
					
						
							|  |  |  |             widget.onEndEditing(); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |         child: UniversalPlatform.isMobile ? _mobile() : _desktop(), | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |       ), | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-11-24 00:25:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |   Widget _mobile() { | 
					
						
							|  |  |  |     return BlocBuilder<CardBloc, CardState>( | 
					
						
							|  |  |  |       builder: (context, state) { | 
					
						
							|  |  |  |         return GestureDetector( | 
					
						
							|  |  |  |           onTap: () => widget.onTap(context), | 
					
						
							|  |  |  |           behavior: HitTestBehavior.opaque, | 
					
						
							|  |  |  |           child: MobileCardContent( | 
					
						
							|  |  |  |             userProfile: widget.userProfile, | 
					
						
							|  |  |  |             rowMeta: state.rowMeta, | 
					
						
							|  |  |  |             cellBuilder: widget.cellBuilder, | 
					
						
							|  |  |  |             styleConfiguration: widget.styleConfiguration, | 
					
						
							|  |  |  |             cells: state.cells, | 
					
						
							|  |  |  |           ), | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |   Widget _desktop() { | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     final accessories = widget.styleConfiguration.showAccessory | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |         ? const <CardAccessory>[ | 
					
						
							|  |  |  |             EditCardAccessory(), | 
					
						
							|  |  |  |             MoreCardOptionsAccessory(), | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |           ] | 
					
						
							|  |  |  |         : null; | 
					
						
							|  |  |  |     return AppFlowyPopover( | 
					
						
							|  |  |  |       controller: popoverController, | 
					
						
							|  |  |  |       triggerActions: PopoverTriggerFlags.none, | 
					
						
							|  |  |  |       constraints: BoxConstraints.loose(const Size(140, 200)), | 
					
						
							|  |  |  |       direction: PopoverDirection.rightWithCenterAligned, | 
					
						
							| 
									
										
										
										
											2024-09-05 12:48:44 +02:00
										 |  |  |       popupBuilder: (_) => RowActionMenu.board( | 
					
						
							|  |  |  |         viewId: _cardBloc.viewId, | 
					
						
							|  |  |  |         rowId: _cardBloc.rowController.rowId, | 
					
						
							|  |  |  |         groupId: widget.groupId, | 
					
						
							|  |  |  |       ), | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |       child: Builder( | 
					
						
							|  |  |  |         builder: (context) { | 
					
						
							|  |  |  |           return RowCardContainer( | 
					
						
							|  |  |  |             buildAccessoryWhen: () => | 
					
						
							|  |  |  |                 !context.watch<CardBloc>().state.isEditing, | 
					
						
							|  |  |  |             accessories: accessories ?? [], | 
					
						
							|  |  |  |             openAccessory: _handleOpenAccessory, | 
					
						
							|  |  |  |             onTap: widget.onTap, | 
					
						
							|  |  |  |             onShiftTap: widget.onShiftTap, | 
					
						
							|  |  |  |             child: BlocBuilder<CardBloc, CardState>( | 
					
						
							|  |  |  |               builder: (context, state) { | 
					
						
							|  |  |  |                 return _CardContent( | 
					
						
							|  |  |  |                   rowMeta: state.rowMeta, | 
					
						
							|  |  |  |                   cellBuilder: widget.cellBuilder, | 
					
						
							|  |  |  |                   styleConfiguration: widget.styleConfiguration, | 
					
						
							|  |  |  |                   cells: state.cells, | 
					
						
							|  |  |  |                   userProfile: widget.userProfile, | 
					
						
							|  |  |  |                   isCompact: widget.isCompact, | 
					
						
							|  |  |  |                 ); | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |           ); | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2022-08-12 20:10:56 +08:00
										 |  |  |       ), | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-19 15:36:59 +08:00
										 |  |  |   void _handleOpenAccessory(AccessoryType newAccessoryType) { | 
					
						
							|  |  |  |     switch (newAccessoryType) { | 
					
						
							|  |  |  |       case AccessoryType.edit: | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |         widget.onStartEditing(); | 
					
						
							| 
									
										
										
										
											2022-09-19 15:36:59 +08:00
										 |  |  |         break; | 
					
						
							|  |  |  |       case AccessoryType.more: | 
					
						
							|  |  |  |         popoverController.show(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-09-18 11:06:04 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | class _CardContent extends StatelessWidget { | 
					
						
							| 
									
										
										
										
											2023-03-08 21:19:44 +08:00
										 |  |  |   const _CardContent({ | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     required this.rowMeta, | 
					
						
							| 
									
										
										
										
											2022-09-18 11:06:04 +08:00
										 |  |  |     required this.cellBuilder, | 
					
						
							|  |  |  |     required this.cells, | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  |     required this.styleConfiguration, | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |     this.userProfile, | 
					
						
							|  |  |  |     this.isCompact = false, | 
					
						
							| 
									
										
										
										
											2023-11-09 00:30:50 +01:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |   final RowMetaPB rowMeta; | 
					
						
							|  |  |  |   final CardCellBuilder cellBuilder; | 
					
						
							| 
									
										
										
										
											2024-05-20 12:49:58 +03:30
										 |  |  |   final List<CellMeta> cells; | 
					
						
							| 
									
										
										
										
											2023-11-09 00:30:50 +01:00
										 |  |  |   final RowCardStyleConfiguration styleConfiguration; | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |   final UserProfilePB? userProfile; | 
					
						
							|  |  |  |   final bool isCompact; | 
					
						
							| 
									
										
										
										
											2023-11-26 15:10:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-18 11:06:04 +08:00
										 |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |     final child = Column( | 
					
						
							| 
									
										
										
										
											2024-09-22 09:34:21 +08:00
										 |  |  |       mainAxisAlignment: MainAxisAlignment.center, | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |       mainAxisSize: MainAxisSize.min, | 
					
						
							|  |  |  |       children: [ | 
					
						
							|  |  |  |         CardCover( | 
					
						
							|  |  |  |           cover: rowMeta.cover, | 
					
						
							|  |  |  |           userProfile: userProfile, | 
					
						
							|  |  |  |           isCompact: isCompact, | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |         Padding( | 
					
						
							|  |  |  |           padding: styleConfiguration.cardPadding, | 
					
						
							|  |  |  |           child: Column( | 
					
						
							| 
									
										
										
										
											2024-09-16 15:18:14 +02:00
										 |  |  |             children: _makeCells(context, rowMeta, cells), | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |           ), | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |       ], | 
					
						
							| 
									
										
										
										
											2022-09-18 11:06:04 +08:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     return styleConfiguration.hoverStyle == null | 
					
						
							|  |  |  |         ? child | 
					
						
							|  |  |  |         : FlowyHover( | 
					
						
							|  |  |  |             style: styleConfiguration.hoverStyle, | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |             buildWhenOnHover: () => !context.read<CardBloc>().state.isEditing, | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |             child: child, | 
					
						
							|  |  |  |           ); | 
					
						
							| 
									
										
										
										
											2022-09-18 11:06:04 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-26 16:07:51 +08:00
										 |  |  |   List<Widget> _makeCells( | 
					
						
							|  |  |  |     BuildContext context, | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     RowMetaPB rowMeta, | 
					
						
							| 
									
										
										
										
											2024-05-20 12:49:58 +03:30
										 |  |  |     List<CellMeta> cells, | 
					
						
							| 
									
										
										
										
											2022-08-26 16:07:51 +08:00
										 |  |  |   ) { | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |     return cells | 
					
						
							|  |  |  |         .mapIndexed( | 
					
						
							|  |  |  |           (int index, CellMeta cellMeta) => _CardContentCell( | 
					
						
							|  |  |  |             cellBuilder: cellBuilder, | 
					
						
							|  |  |  |             cellMeta: cellMeta, | 
					
						
							|  |  |  |             rowMeta: rowMeta, | 
					
						
							|  |  |  |             isTitle: index == 0, | 
					
						
							|  |  |  |             styleMap: styleConfiguration.cellStyleMap, | 
					
						
							|  |  |  |           ), | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         .toList(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _CardContentCell extends StatefulWidget { | 
					
						
							|  |  |  |   const _CardContentCell({ | 
					
						
							|  |  |  |     required this.cellBuilder, | 
					
						
							|  |  |  |     required this.cellMeta, | 
					
						
							|  |  |  |     required this.rowMeta, | 
					
						
							|  |  |  |     required this.isTitle, | 
					
						
							|  |  |  |     required this.styleMap, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   final CellMeta cellMeta; | 
					
						
							|  |  |  |   final RowMetaPB rowMeta; | 
					
						
							|  |  |  |   final CardCellBuilder cellBuilder; | 
					
						
							|  |  |  |   final CardCellStyleMap styleMap; | 
					
						
							|  |  |  |   final bool isTitle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   State<_CardContentCell> createState() => _CardContentCellState(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _CardContentCellState extends State<_CardContentCell> { | 
					
						
							|  |  |  |   late final EditableCardNotifier? cellNotifier; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   void initState() { | 
					
						
							|  |  |  |     super.initState(); | 
					
						
							|  |  |  |     cellNotifier = widget.isTitle ? EditableCardNotifier() : null; | 
					
						
							|  |  |  |     cellNotifier?.isCellEditing.addListener(listener); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void listener() { | 
					
						
							|  |  |  |     final isEditing = cellNotifier!.isCellEditing.value; | 
					
						
							|  |  |  |     context.read<CardBloc>().add(CardEvent.setIsEditing(isEditing)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   void dispose() { | 
					
						
							| 
									
										
										
										
											2024-11-04 03:26:48 +01:00
										 |  |  |     cellNotifier?.isCellEditing.removeListener(listener); | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |     cellNotifier?.dispose(); | 
					
						
							|  |  |  |     super.dispose(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							|  |  |  |     return BlocListener<CardBloc, CardState>( | 
					
						
							|  |  |  |       listenWhen: (previous, current) => | 
					
						
							|  |  |  |           previous.isEditing != current.isEditing, | 
					
						
							|  |  |  |       listener: (context, state) { | 
					
						
							|  |  |  |         cellNotifier?.isCellEditing.value = state.isEditing; | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       child: widget.cellBuilder.build( | 
					
						
							|  |  |  |         cellContext: widget.cellMeta.cellContext(), | 
					
						
							|  |  |  |         styleMap: widget.styleMap, | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |         cellNotifier: cellNotifier, | 
					
						
							| 
									
										
										
										
											2024-09-29 11:34:45 +08:00
										 |  |  |         hasNotes: !widget.rowMeta.isDocumentEmpty, | 
					
						
							|  |  |  |       ), | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2022-08-10 17:59:28 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-13 11:51:26 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  | class CardCover extends StatelessWidget { | 
					
						
							|  |  |  |   const CardCover({ | 
					
						
							|  |  |  |     super.key, | 
					
						
							|  |  |  |     this.cover, | 
					
						
							|  |  |  |     this.userProfile, | 
					
						
							|  |  |  |     this.isCompact = false, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   final RowCoverPB? cover; | 
					
						
							|  |  |  |   final UserProfilePB? userProfile; | 
					
						
							|  |  |  |   final bool isCompact; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							|  |  |  |     if (cover == null || | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  |         cover!.data.isEmpty || | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |         cover!.uploadType == FileUploadTypePB.CloudFile && | 
					
						
							|  |  |  |             userProfile == null) { | 
					
						
							|  |  |  |       return const SizedBox.shrink(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return Container( | 
					
						
							|  |  |  |       clipBehavior: Clip.antiAlias, | 
					
						
							|  |  |  |       decoration: BoxDecoration( | 
					
						
							|  |  |  |         borderRadius: const BorderRadius.only( | 
					
						
							|  |  |  |           topLeft: Radius.circular(4), | 
					
						
							|  |  |  |           topRight: Radius.circular(4), | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |         color: Theme.of(context).cardColor, | 
					
						
							|  |  |  |       ), | 
					
						
							|  |  |  |       child: Row( | 
					
						
							|  |  |  |         children: [ | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  |           Expanded(child: _renderCover(context, cover!)), | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  |         ], | 
					
						
							|  |  |  |       ), | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-09-23 16:28:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   Widget _renderCover(BuildContext context, RowCoverPB cover) { | 
					
						
							|  |  |  |     final height = isCompact ? 50.0 : 100.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cover.coverType == CoverTypePB.FileCover) { | 
					
						
							|  |  |  |       return SizedBox( | 
					
						
							|  |  |  |         height: height, | 
					
						
							|  |  |  |         width: double.infinity, | 
					
						
							|  |  |  |         child: AFImage( | 
					
						
							|  |  |  |           url: cover.data, | 
					
						
							|  |  |  |           uploadType: cover.uploadType, | 
					
						
							|  |  |  |           userProfile: userProfile, | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cover.coverType == CoverTypePB.AssetCover) { | 
					
						
							|  |  |  |       return SizedBox( | 
					
						
							|  |  |  |         height: height, | 
					
						
							|  |  |  |         width: double.infinity, | 
					
						
							|  |  |  |         child: Image.asset( | 
					
						
							|  |  |  |           PageStyleCoverImageType.builtInImagePath(cover.data), | 
					
						
							|  |  |  |           fit: BoxFit.cover, | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cover.coverType == CoverTypePB.ColorCover) { | 
					
						
							|  |  |  |       final color = FlowyTint.fromId(cover.data)?.color(context) ?? | 
					
						
							|  |  |  |           cover.data.tryToColor(); | 
					
						
							|  |  |  |       return Container( | 
					
						
							|  |  |  |         height: height, | 
					
						
							|  |  |  |         width: double.infinity, | 
					
						
							|  |  |  |         color: color, | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cover.coverType == CoverTypePB.GradientCover) { | 
					
						
							|  |  |  |       return Container( | 
					
						
							|  |  |  |         height: height, | 
					
						
							|  |  |  |         width: double.infinity, | 
					
						
							|  |  |  |         decoration: BoxDecoration( | 
					
						
							|  |  |  |           gradient: FlowyGradientColor.fromId(cover.data).linear, | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return const SizedBox.shrink(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-09-15 22:33:18 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  | class EditCardAccessory extends StatelessWidget with CardAccessory { | 
					
						
							|  |  |  |   const EditCardAccessory({super.key}); | 
					
						
							| 
									
										
										
										
											2022-08-13 11:51:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							| 
									
										
										
										
											2022-08-30 20:54:11 +08:00
										 |  |  |     return Padding( | 
					
						
							|  |  |  |       padding: const EdgeInsets.all(3.0), | 
					
						
							| 
									
										
										
										
											2023-08-14 13:34:01 -07:00
										 |  |  |       child: FlowySvg( | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |         FlowySvgs.edit_s, | 
					
						
							| 
									
										
										
										
											2023-11-24 00:25:12 +08:00
										 |  |  |         color: Theme.of(context).hintColor, | 
					
						
							| 
									
										
										
										
											2022-11-10 14:22:18 +08:00
										 |  |  |       ), | 
					
						
							| 
									
										
										
										
											2022-08-30 20:54:11 +08:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2022-08-13 11:51:26 +08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |   AccessoryType get type => AccessoryType.edit; | 
					
						
							| 
									
										
										
										
											2022-08-13 11:51:26 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-08-30 20:54:11 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  | class MoreCardOptionsAccessory extends StatelessWidget with CardAccessory { | 
					
						
							|  |  |  |   const MoreCardOptionsAccessory({super.key}); | 
					
						
							| 
									
										
										
										
											2022-08-30 20:54:11 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							|  |  |  |     return Padding( | 
					
						
							|  |  |  |       padding: const EdgeInsets.all(3.0), | 
					
						
							| 
									
										
										
										
											2023-08-14 13:34:01 -07:00
										 |  |  |       child: FlowySvg( | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |         FlowySvgs.three_dots_s, | 
					
						
							| 
									
										
										
										
											2023-11-24 00:25:12 +08:00
										 |  |  |         color: Theme.of(context).hintColor, | 
					
						
							| 
									
										
										
										
											2022-08-30 20:54:11 +08:00
										 |  |  |       ), | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							| 
									
										
										
										
											2024-05-10 10:02:10 +08:00
										 |  |  |   AccessoryType get type => AccessoryType.more; | 
					
						
							| 
									
										
										
										
											2022-08-30 20:54:11 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class RowCardStyleConfiguration { | 
					
						
							|  |  |  |   const RowCardStyleConfiguration({ | 
					
						
							| 
									
										
										
										
											2024-01-24 23:59:45 +08:00
										 |  |  |     required this.cellStyleMap, | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  |     this.showAccessory = true, | 
					
						
							| 
									
										
										
										
											2025-03-04 11:29:38 +08:00
										 |  |  |     this.cardPadding = const EdgeInsets.all(4), | 
					
						
							| 
									
										
										
										
											2023-05-31 16:52:37 +08:00
										 |  |  |     this.hoverStyle, | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2024-01-25 16:37:36 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   final CardCellStyleMap cellStyleMap; | 
					
						
							|  |  |  |   final bool showAccessory; | 
					
						
							|  |  |  |   final EdgeInsets cardPadding; | 
					
						
							|  |  |  |   final HoverStyle? hoverStyle; | 
					
						
							| 
									
										
										
										
											2023-04-30 20:50:24 +08:00
										 |  |  | } |