123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542 |
- import 'dart:collection';
- import 'package:app_flowy/plugins/grid/application/field/grid_listener.dart';
- import 'package:app_flowy/plugins/grid/application/filter/filter_listener.dart';
- import 'package:app_flowy/plugins/grid/application/filter/filter_service.dart';
- import 'package:app_flowy/plugins/grid/application/grid_service.dart';
- import 'package:app_flowy/plugins/grid/application/setting/setting_listener.dart';
- import 'package:app_flowy/plugins/grid/application/setting/setting_service.dart';
- import 'package:app_flowy/plugins/grid/presentation/widgets/filter/filter_info.dart';
- import 'package:dartz/dartz.dart';
- import 'package:flowy_sdk/log.dart';
- import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
- import 'package:flowy_sdk/protobuf/flowy-grid/field_entities.pb.dart';
- import 'package:flowy_sdk/protobuf/flowy-grid/group.pb.dart';
- import 'package:flowy_sdk/protobuf/flowy-grid/setting_entities.pb.dart';
- import 'package:flowy_sdk/protobuf/flowy-grid/util.pb.dart';
- import 'package:flutter/foundation.dart';
- import '../row/row_cache.dart';
- class _GridFieldNotifier extends ChangeNotifier {
- List<FieldInfo> _fieldInfos = [];
- set fieldInfos(List<FieldInfo> fieldInfos) {
- _fieldInfos = fieldInfos;
- notifyListeners();
- }
- void notify() {
- notifyListeners();
- }
- List<FieldInfo> get fieldInfos => _fieldInfos;
- }
- class _GridFilterNotifier extends ChangeNotifier {
- List<FilterInfo> _filters = [];
- set filters(List<FilterInfo> filters) {
- _filters = filters;
- notifyListeners();
- }
- void notify() {
- notifyListeners();
- }
- List<FilterInfo> get filters => _filters;
- }
- typedef OnReceiveUpdateFields = void Function(List<FieldInfo>);
- typedef OnReceiveFields = void Function(List<FieldInfo>);
- typedef OnReceiveFilters = void Function(List<FilterInfo>);
- class GridFieldController {
- final String gridId;
- // Listeners
- final GridFieldsListener _fieldListener;
- final SettingListener _settingListener;
- final FiltersListener _filterListener;
- // FFI services
- final GridFFIService _gridFFIService;
- final SettingFFIService _settingFFIService;
- final FilterFFIService _filterFFIService;
- // Field callbacks
- final Map<OnReceiveFields, VoidCallback> _fieldCallbacks = {};
- _GridFieldNotifier? _fieldNotifier = _GridFieldNotifier();
- // Field updated callbacks
- final Map<OnReceiveUpdateFields, void Function(List<FieldInfo>)>
- _updatedFieldCallbacks = {};
- // Group callbacks
- final Map<String, GroupConfigurationPB> _groupConfigurationByFieldId = {};
- // Filter callbacks
- final Map<OnReceiveFilters, VoidCallback> _filterCallbacks = {};
- _GridFilterNotifier? _filterNotifier = _GridFilterNotifier();
- final Map<String, FilterPB> _filterPBByFieldId = {};
- // Getters
- List<FieldInfo> get fieldInfos => [..._fieldNotifier?.fieldInfos ?? []];
- List<FilterInfo> get filterInfos => [..._filterNotifier?.filters ?? []];
- FieldInfo? getField(String fieldId) {
- final fields = _fieldNotifier?.fieldInfos
- .where((element) => element.id == fieldId)
- .toList() ??
- [];
- if (fields.isEmpty) {
- return null;
- }
- assert(fields.length == 1);
- return fields.first;
- }
- FilterInfo? getFilter(String filterId) {
- final filters = _filterNotifier?.filters
- .where((element) => element.filter.id == filterId)
- .toList() ??
- [];
- if (filters.isEmpty) {
- return null;
- }
- assert(filters.length == 1);
- return filters.first;
- }
- GridFieldController({required this.gridId})
- : _fieldListener = GridFieldsListener(gridId: gridId),
- _settingListener = SettingListener(gridId: gridId),
- _filterListener = FiltersListener(viewId: gridId),
- _gridFFIService = GridFFIService(gridId: gridId),
- _filterFFIService = FilterFFIService(viewId: gridId),
- _settingFFIService = SettingFFIService(viewId: gridId) {
- //Listen on field's changes
- _listenOnFieldChanges();
- //Listen on setting changes
- _listenOnSettingChanges();
- //Listen on the fitler changes
- _listenOnFilterChanges();
- _settingFFIService.getSetting().then((result) {
- result.fold(
- (setting) => _updateSettingConfiguration(setting),
- (err) => Log.error(err),
- );
- });
- }
- void _listenOnFilterChanges() {
- //Listen on the fitler changes
- _filterListener.start(onFilterChanged: (result) {
- result.fold(
- (changeset) {
- final List<FilterInfo> filters = filterInfos;
- // Deletes the filters
- final deleteFilterIds =
- changeset.deleteFilters.map((e) => e.id).toList();
- if (deleteFilterIds.isNotEmpty) {
- filters.retainWhere(
- (element) => !deleteFilterIds.contains(element.filter.id),
- );
- _filterPBByFieldId.removeWhere(
- (key, value) => deleteFilterIds.contains(value.id));
- }
- // Inserts the new filter if it's not exist
- for (final newFilter in changeset.insertFilters) {
- final filterIndex = filters
- .indexWhere((element) => element.filter.id == newFilter.id);
- if (filterIndex == -1) {
- final fieldInfo = _findFieldInfoForFilter(fieldInfos, newFilter);
- if (fieldInfo != null) {
- _filterPBByFieldId[fieldInfo.id] = newFilter;
- filters.add(FilterInfo(gridId, newFilter, fieldInfo));
- }
- }
- }
- for (final updatedFilter in changeset.updateFilters) {
- final filterIndex = filters.indexWhere(
- (element) => element.filter.id == updatedFilter.filterId,
- );
- // Remove the old filter
- if (filterIndex != -1) {
- filters.removeAt(filterIndex);
- _filterPBByFieldId.removeWhere(
- (key, value) => value.id == updatedFilter.filterId);
- }
- // Insert the filter if there is a fitler and its field info is
- // not null
- if (updatedFilter.hasFilter()) {
- final fieldInfo = _findFieldInfoForFilter(
- fieldInfos,
- updatedFilter.filter,
- );
- if (fieldInfo != null) {
- // Insert the filter with the position: filterIndex, otherwise,
- // append it to the end of the list.
- final filterInfo =
- FilterInfo(gridId, updatedFilter.filter, fieldInfo);
- if (filterIndex != -1) {
- filters.insert(filterIndex, filterInfo);
- } else {
- filters.add(filterInfo);
- }
- _filterPBByFieldId[fieldInfo.id] = updatedFilter.filter;
- }
- }
- }
- _updateFieldInfos();
- _filterNotifier?.filters = filters;
- },
- (err) => Log.error(err),
- );
- });
- }
- void _listenOnSettingChanges() {
- //Listen on setting changes
- _settingListener.start(onSettingUpdated: (result) {
- result.fold(
- (setting) => _updateSettingConfiguration(setting),
- (r) => Log.error(r),
- );
- });
- }
- void _listenOnFieldChanges() {
- //Listen on field's changes
- _fieldListener.start(onFieldsChanged: (result) {
- result.fold(
- (changeset) {
- _deleteFields(changeset.deletedFields);
- _insertFields(changeset.insertedFields);
- final updateFields = _updateFields(changeset.updatedFields);
- for (final listener in _updatedFieldCallbacks.values) {
- listener(updateFields);
- }
- },
- (err) => Log.error(err),
- );
- });
- }
- void _updateSettingConfiguration(GridSettingPB setting) {
- _groupConfigurationByFieldId.clear();
- for (final configuration in setting.groupConfigurations.items) {
- _groupConfigurationByFieldId[configuration.fieldId] = configuration;
- }
- for (final configuration in setting.filters.items) {
- _filterPBByFieldId[configuration.fieldId] = configuration;
- }
- _updateFieldInfos();
- }
- void _updateFieldInfos() {
- if (_fieldNotifier != null) {
- for (var field in _fieldNotifier!.fieldInfos) {
- field._isGroupField = _groupConfigurationByFieldId[field.id] != null;
- field._hasFilter = _filterPBByFieldId[field.id] != null;
- }
- _fieldNotifier?.notify();
- }
- }
- Future<void> dispose() async {
- await _fieldListener.stop();
- await _filterListener.stop();
- await _settingListener.stop();
- for (final callback in _fieldCallbacks.values) {
- _fieldNotifier?.removeListener(callback);
- }
- _fieldNotifier?.dispose();
- _fieldNotifier = null;
- for (final callback in _filterCallbacks.values) {
- _filterNotifier?.removeListener(callback);
- }
- _filterNotifier?.dispose();
- _filterNotifier = null;
- }
- Future<Either<Unit, FlowyError>> loadFields({
- required List<FieldIdPB> fieldIds,
- }) async {
- final result = await _gridFFIService.getFields(fieldIds: fieldIds);
- return Future(
- () => result.fold(
- (newFields) {
- _fieldNotifier?.fieldInfos =
- newFields.map((field) => FieldInfo(field: field)).toList();
- _loadFilters();
- _updateFieldInfos();
- return left(unit);
- },
- (err) => right(err),
- ),
- );
- }
- Future<Either<Unit, FlowyError>> _loadFilters() async {
- return _filterFFIService.getAllFilters().then((result) {
- return result.fold(
- (filterPBs) {
- final List<FilterInfo> filters = [];
- for (final filterPB in filterPBs) {
- final fieldInfo = _findFieldInfoForFilter(fieldInfos, filterPB);
- if (fieldInfo != null) {
- final filterInfo = FilterInfo(gridId, filterPB, fieldInfo);
- filters.add(filterInfo);
- }
- }
- _updateFieldInfos();
- _filterNotifier?.filters = filters;
- return left(unit);
- },
- (err) => right(err),
- );
- });
- }
- void addListener({
- OnReceiveFields? onFields,
- OnReceiveUpdateFields? onFieldsUpdated,
- OnReceiveFilters? onFilters,
- bool Function()? listenWhen,
- }) {
- if (onFieldsUpdated != null) {
- callback(List<FieldInfo> updateFields) {
- if (listenWhen != null && listenWhen() == false) {
- return;
- }
- onFieldsUpdated(updateFields);
- }
- _updatedFieldCallbacks[onFieldsUpdated] = callback;
- }
- if (onFields != null) {
- callback() {
- if (listenWhen != null && listenWhen() == false) {
- return;
- }
- onFields(fieldInfos);
- }
- _fieldCallbacks[onFields] = callback;
- _fieldNotifier?.addListener(callback);
- }
- if (onFilters != null) {
- callback() {
- if (listenWhen != null && listenWhen() == false) {
- return;
- }
- onFilters(filterInfos);
- }
- _filterCallbacks[onFilters] = callback;
- _filterNotifier?.addListener(callback);
- }
- }
- void removeListener({
- OnReceiveFields? onFieldsListener,
- OnReceiveFilters? onFiltersListener,
- OnReceiveUpdateFields? onChangesetListener,
- }) {
- if (onFieldsListener != null) {
- final callback = _fieldCallbacks.remove(onFieldsListener);
- if (callback != null) {
- _fieldNotifier?.removeListener(callback);
- }
- }
- if (onFiltersListener != null) {
- final callback = _filterCallbacks.remove(onFiltersListener);
- if (callback != null) {
- _filterNotifier?.removeListener(callback);
- }
- }
- }
- void _deleteFields(List<FieldIdPB> deletedFields) {
- if (deletedFields.isEmpty) {
- return;
- }
- final List<FieldInfo> newFields = fieldInfos;
- final Map<String, FieldIdPB> deletedFieldMap = {
- for (var fieldOrder in deletedFields) fieldOrder.fieldId: fieldOrder
- };
- newFields.retainWhere((field) => (deletedFieldMap[field.id] == null));
- _fieldNotifier?.fieldInfos = newFields;
- }
- void _insertFields(List<IndexFieldPB> insertedFields) {
- if (insertedFields.isEmpty) {
- return;
- }
- final List<FieldInfo> newFields = fieldInfos;
- for (final indexField in insertedFields) {
- final gridField = FieldInfo(field: indexField.field_1);
- if (newFields.length > indexField.index) {
- newFields.insert(indexField.index, gridField);
- } else {
- newFields.add(gridField);
- }
- }
- _fieldNotifier?.fieldInfos = newFields;
- }
- List<FieldInfo> _updateFields(List<FieldPB> updatedFieldPBs) {
- if (updatedFieldPBs.isEmpty) {
- return [];
- }
- final List<FieldInfo> newFields = fieldInfos;
- final List<FieldInfo> updatedFields = [];
- for (final updatedFieldPB in updatedFieldPBs) {
- final index =
- newFields.indexWhere((field) => field.id == updatedFieldPB.id);
- if (index != -1) {
- newFields.removeAt(index);
- final fieldInfo = FieldInfo(field: updatedFieldPB);
- newFields.insert(index, fieldInfo);
- updatedFields.add(fieldInfo);
- }
- }
- if (updatedFields.isNotEmpty) {
- _fieldNotifier?.fieldInfos = newFields;
- }
- return updatedFields;
- }
- }
- class GridRowFieldNotifierImpl extends IGridRowFieldNotifier {
- final GridFieldController _cache;
- OnReceiveUpdateFields? _onChangesetFn;
- OnReceiveFields? _onFieldFn;
- GridRowFieldNotifierImpl(GridFieldController cache) : _cache = cache;
- @override
- UnmodifiableListView<FieldInfo> get fields =>
- UnmodifiableListView(_cache.fieldInfos);
- @override
- void onRowFieldsChanged(VoidCallback callback) {
- _onFieldFn = (_) => callback();
- _cache.addListener(onFields: _onFieldFn);
- }
- @override
- void onRowFieldChanged(void Function(FieldInfo) callback) {
- _onChangesetFn = (List<FieldInfo> fieldInfos) {
- for (final updatedField in fieldInfos) {
- callback(updatedField);
- }
- };
- _cache.addListener(onFieldsUpdated: _onChangesetFn);
- }
- @override
- void onRowDispose() {
- if (_onFieldFn != null) {
- _cache.removeListener(onFieldsListener: _onFieldFn!);
- _onFieldFn = null;
- }
- if (_onChangesetFn != null) {
- _cache.removeListener(onChangesetListener: _onChangesetFn!);
- _onChangesetFn = null;
- }
- }
- }
- FieldInfo? _findFieldInfoForFilter(
- List<FieldInfo> fieldInfos, FilterPB filter) {
- final fieldIndex = fieldInfos.indexWhere((element) {
- return element.id == filter.fieldId &&
- element.fieldType == filter.fieldType;
- });
- if (fieldIndex != -1) {
- return fieldInfos[fieldIndex];
- } else {
- return null;
- }
- }
- class FieldInfo {
- final FieldPB _field;
- bool _isGroupField = false;
- bool _hasFilter = false;
- String get id => _field.id;
- FieldType get fieldType => _field.fieldType;
- bool get visibility => _field.visibility;
- double get width => _field.width.toDouble();
- bool get isPrimary => _field.isPrimary;
- String get name => _field.name;
- FieldPB get field => _field;
- bool get isGroupField => _isGroupField;
- bool get hasFilter => _hasFilter;
- bool get canGroup {
- switch (_field.fieldType) {
- case FieldType.Checkbox:
- return true;
- case FieldType.DateTime:
- return false;
- case FieldType.MultiSelect:
- return true;
- case FieldType.Number:
- return false;
- case FieldType.RichText:
- return false;
- case FieldType.SingleSelect:
- return true;
- case FieldType.URL:
- return false;
- }
- return false;
- }
- bool get canCreateFilter {
- if (hasFilter) return false;
- switch (_field.fieldType) {
- case FieldType.Checkbox:
- case FieldType.MultiSelect:
- case FieldType.RichText:
- case FieldType.SingleSelect:
- return true;
- default:
- return false;
- }
- }
- FieldInfo({required FieldPB field}) : _field = field;
- }
|