mirror of
https://github.com/AppFlowy-IO/AppFlowy.git
synced 2025-10-24 06:24:36 +00:00

* feat: refactor databse styles * feat: support compact mode for databse * feat: support dynamic height for board * fix: add reference icon for database view in document * feat: support data sync for database node in document * fix: add hover effect in compact mode switcher * fix: title of document not align correctly with a large screen * fix: some launch review issues * fix: auto hide the Hidden Groups unless the user clicks it to show * fix: testing error * chore: update board version * chore: update database menu buttons * fix: some launch review issues --------- Co-authored-by: Lucas <lucas.xu@appflowy.io>
472 lines
13 KiB
Dart
472 lines
13 KiB
Dart
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
|
import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart';
|
|
import 'package:appflowy/mobile/presentation/database/card/card.dart';
|
|
import 'package:appflowy/plugins/database/application/field/field_controller.dart';
|
|
import 'package:appflowy/plugins/database/application/row/row_cache.dart';
|
|
import 'package:appflowy/plugins/database/application/row/row_controller.dart';
|
|
import 'package:appflowy/plugins/database/grid/presentation/widgets/row/action.dart';
|
|
import 'package:appflowy/shared/af_image.dart';
|
|
import 'package:appflowy/shared/flowy_gradient_colors.dart';
|
|
import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
|
|
import 'package:appflowy_backend/protobuf/flowy-user/user_profile.pb.dart';
|
|
import 'package:appflowy_editor/appflowy_editor.dart';
|
|
import 'package:collection/collection.dart';
|
|
import 'package:flowy_infra/theme_extension.dart';
|
|
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
|
|
import 'package:flowy_infra_ui/style_widget/hover.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
import 'package:universal_platform/universal_platform.dart';
|
|
|
|
import '../cell/card_cell_builder.dart';
|
|
import '../cell/card_cell_skeleton/card_cell.dart';
|
|
import 'card_bloc.dart';
|
|
import 'container/accessory.dart';
|
|
import 'container/card_container.dart';
|
|
|
|
/// Edit a database row with card style widget
|
|
class RowCard extends StatefulWidget {
|
|
const RowCard({
|
|
super.key,
|
|
required this.fieldController,
|
|
required this.rowMeta,
|
|
required this.viewId,
|
|
required this.isEditing,
|
|
required this.rowCache,
|
|
required this.cellBuilder,
|
|
required this.onTap,
|
|
required this.onStartEditing,
|
|
required this.onEndEditing,
|
|
required this.styleConfiguration,
|
|
this.onShiftTap,
|
|
this.groupingFieldId,
|
|
this.groupId,
|
|
required this.userProfile,
|
|
this.isCompact = false,
|
|
});
|
|
|
|
final FieldController fieldController;
|
|
final RowMetaPB rowMeta;
|
|
final String viewId;
|
|
final String? groupingFieldId;
|
|
final String? groupId;
|
|
|
|
final bool isEditing;
|
|
final RowCache rowCache;
|
|
|
|
/// The [CardCellBuilder] is used to build the card cells.
|
|
final CardCellBuilder cellBuilder;
|
|
|
|
/// Called when the user taps on the card.
|
|
final void Function(BuildContext context) onTap;
|
|
|
|
final void Function(BuildContext context)? onShiftTap;
|
|
|
|
/// Called when the user starts editing the card.
|
|
final VoidCallback onStartEditing;
|
|
|
|
/// Called when the user ends editing the card.
|
|
final VoidCallback onEndEditing;
|
|
|
|
final RowCardStyleConfiguration styleConfiguration;
|
|
|
|
/// 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;
|
|
|
|
@override
|
|
State<RowCard> createState() => _RowCardState();
|
|
}
|
|
|
|
class _RowCardState extends State<RowCard> {
|
|
final popoverController = PopoverController();
|
|
late final CardBloc _cardBloc;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
final rowController = RowController(
|
|
viewId: widget.viewId,
|
|
rowMeta: widget.rowMeta,
|
|
rowCache: widget.rowCache,
|
|
);
|
|
|
|
_cardBloc = CardBloc(
|
|
fieldController: widget.fieldController,
|
|
viewId: widget.viewId,
|
|
groupFieldId: widget.groupingFieldId,
|
|
isEditing: widget.isEditing,
|
|
rowController: rowController,
|
|
)..add(const CardEvent.initial());
|
|
}
|
|
|
|
@override
|
|
void didUpdateWidget(covariant oldWidget) {
|
|
if (widget.isEditing != _cardBloc.state.isEditing) {
|
|
_cardBloc.add(CardEvent.setIsEditing(widget.isEditing));
|
|
}
|
|
super.didUpdateWidget(oldWidget);
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_cardBloc.close();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return BlocProvider.value(
|
|
value: _cardBloc,
|
|
child: BlocListener<CardBloc, CardState>(
|
|
listenWhen: (previous, current) =>
|
|
previous.isEditing != current.isEditing,
|
|
listener: (context, state) {
|
|
if (!state.isEditing) {
|
|
widget.onEndEditing();
|
|
}
|
|
},
|
|
child: UniversalPlatform.isMobile ? _mobile() : _desktop(),
|
|
),
|
|
);
|
|
}
|
|
|
|
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,
|
|
),
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
Widget _desktop() {
|
|
final accessories = widget.styleConfiguration.showAccessory
|
|
? const <CardAccessory>[
|
|
EditCardAccessory(),
|
|
MoreCardOptionsAccessory(),
|
|
]
|
|
: null;
|
|
return AppFlowyPopover(
|
|
controller: popoverController,
|
|
triggerActions: PopoverTriggerFlags.none,
|
|
constraints: BoxConstraints.loose(const Size(140, 200)),
|
|
direction: PopoverDirection.rightWithCenterAligned,
|
|
popupBuilder: (_) => RowActionMenu.board(
|
|
viewId: _cardBloc.viewId,
|
|
rowId: _cardBloc.rowController.rowId,
|
|
groupId: widget.groupId,
|
|
),
|
|
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,
|
|
);
|
|
},
|
|
),
|
|
);
|
|
},
|
|
),
|
|
);
|
|
}
|
|
|
|
void _handleOpenAccessory(AccessoryType newAccessoryType) {
|
|
switch (newAccessoryType) {
|
|
case AccessoryType.edit:
|
|
widget.onStartEditing();
|
|
break;
|
|
case AccessoryType.more:
|
|
popoverController.show();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
class _CardContent extends StatelessWidget {
|
|
const _CardContent({
|
|
required this.rowMeta,
|
|
required this.cellBuilder,
|
|
required this.cells,
|
|
required this.styleConfiguration,
|
|
this.userProfile,
|
|
this.isCompact = false,
|
|
});
|
|
|
|
final RowMetaPB rowMeta;
|
|
final CardCellBuilder cellBuilder;
|
|
final List<CellMeta> cells;
|
|
final RowCardStyleConfiguration styleConfiguration;
|
|
final UserProfilePB? userProfile;
|
|
final bool isCompact;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final child = Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
CardCover(
|
|
cover: rowMeta.cover,
|
|
userProfile: userProfile,
|
|
isCompact: isCompact,
|
|
),
|
|
Padding(
|
|
padding: styleConfiguration.cardPadding,
|
|
child: Column(
|
|
children: _makeCells(context, rowMeta, cells),
|
|
),
|
|
),
|
|
],
|
|
);
|
|
return styleConfiguration.hoverStyle == null
|
|
? child
|
|
: FlowyHover(
|
|
style: styleConfiguration.hoverStyle,
|
|
buildWhenOnHover: () => !context.read<CardBloc>().state.isEditing,
|
|
child: child,
|
|
);
|
|
}
|
|
|
|
List<Widget> _makeCells(
|
|
BuildContext context,
|
|
RowMetaPB rowMeta,
|
|
List<CellMeta> cells,
|
|
) {
|
|
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() {
|
|
cellNotifier?.isCellEditing.removeListener(listener);
|
|
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,
|
|
cellNotifier: cellNotifier,
|
|
hasNotes: !widget.rowMeta.isDocumentEmpty,
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
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 ||
|
|
cover!.data.isEmpty ||
|
|
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: [
|
|
Expanded(child: _renderCover(context, cover!)),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
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();
|
|
}
|
|
}
|
|
|
|
class EditCardAccessory extends StatelessWidget with CardAccessory {
|
|
const EditCardAccessory({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Padding(
|
|
padding: const EdgeInsets.all(3.0),
|
|
child: FlowySvg(
|
|
FlowySvgs.edit_s,
|
|
color: Theme.of(context).hintColor,
|
|
),
|
|
);
|
|
}
|
|
|
|
@override
|
|
AccessoryType get type => AccessoryType.edit;
|
|
}
|
|
|
|
class MoreCardOptionsAccessory extends StatelessWidget with CardAccessory {
|
|
const MoreCardOptionsAccessory({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Padding(
|
|
padding: const EdgeInsets.all(3.0),
|
|
child: FlowySvg(
|
|
FlowySvgs.three_dots_s,
|
|
color: Theme.of(context).hintColor,
|
|
),
|
|
);
|
|
}
|
|
|
|
@override
|
|
AccessoryType get type => AccessoryType.more;
|
|
}
|
|
|
|
class RowCardStyleConfiguration {
|
|
const RowCardStyleConfiguration({
|
|
required this.cellStyleMap,
|
|
this.showAccessory = true,
|
|
this.cardPadding = const EdgeInsets.all(4),
|
|
this.hoverStyle,
|
|
});
|
|
|
|
final CardCellStyleMap cellStyleMap;
|
|
final bool showAccessory;
|
|
final EdgeInsets cardPadding;
|
|
final HoverStyle? hoverStyle;
|
|
}
|