123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286 |
- use crate::entities::{GroupChangesetPB, GroupViewChangesetPB, InsertedRowPB, RowPB};
- use crate::services::cell::{decode_any_cell_data, CellBytesParser};
- use crate::services::group::action::GroupAction;
- use crate::services::group::configuration::GroupContext;
- use crate::services::group::entities::Group;
- use flowy_error::FlowyResult;
- use flowy_grid_data_model::revision::{
- FieldRevision, GroupConfigurationContentSerde, GroupRevision, RowChangeset, RowRevision, TypeOptionDataDeserializer,
- };
- use std::marker::PhantomData;
- use std::sync::Arc;
- // Each kind of group must implement this trait to provide custom group
- // operations. For example, insert cell data to the row_rev when creating
- // a new row.
- pub trait GroupController: GroupControllerSharedOperation + Send + Sync {
- fn will_create_row(&mut self, row_rev: &mut RowRevision, field_rev: &FieldRevision, group_id: &str);
- }
- pub trait GroupGenerator {
- type Context;
- type TypeOptionType;
- fn generate_groups(
- field_id: &str,
- group_ctx: &Self::Context,
- type_option: &Option<Self::TypeOptionType>,
- ) -> Vec<GeneratedGroup>;
- }
- pub struct GeneratedGroup {
- pub group_rev: GroupRevision,
- pub filter_content: String,
- }
- pub struct MoveGroupRowContext<'a> {
- pub row_rev: &'a RowRevision,
- pub row_changeset: &'a mut RowChangeset,
- pub field_rev: &'a FieldRevision,
- pub to_group_id: &'a str,
- pub to_row_id: Option<String>,
- }
- // Defines the shared actions each group controller can perform.
- pub trait GroupControllerSharedOperation: Send + Sync {
- // The field that is used for grouping the rows
- fn field_id(&self) -> &str;
- fn groups(&self) -> Vec<Group>;
- fn get_group(&self, group_id: &str) -> Option<(usize, Group)>;
- fn fill_groups(&mut self, row_revs: &[Arc<RowRevision>], field_rev: &FieldRevision) -> FlowyResult<()>;
- fn move_group(&mut self, from_group_id: &str, to_group_id: &str) -> FlowyResult<()>;
- fn did_update_row(
- &mut self,
- row_rev: &RowRevision,
- field_rev: &FieldRevision,
- ) -> FlowyResult<Vec<GroupChangesetPB>>;
- fn did_delete_row(
- &mut self,
- row_rev: &RowRevision,
- field_rev: &FieldRevision,
- ) -> FlowyResult<Vec<GroupChangesetPB>>;
- fn move_group_row(&mut self, context: MoveGroupRowContext) -> FlowyResult<Vec<GroupChangesetPB>>;
- fn did_update_field(&mut self, field_rev: &FieldRevision) -> FlowyResult<Option<GroupViewChangesetPB>>;
- }
- /// C: represents the group configuration that impl [GroupConfigurationSerde]
- /// T: the type option data deserializer that impl [TypeOptionDataDeserializer]
- /// G: the group generator, [GroupGenerator]
- /// P: the parser that impl [CellBytesParser] for the CellBytes
- pub struct GenericGroupController<C, T, G, P> {
- pub field_id: String,
- pub type_option: Option<T>,
- pub group_ctx: GroupContext<C>,
- group_action_phantom: PhantomData<G>,
- cell_parser_phantom: PhantomData<P>,
- }
- impl<C, T, G, P> GenericGroupController<C, T, G, P>
- where
- C: GroupConfigurationContentSerde,
- T: TypeOptionDataDeserializer,
- G: GroupGenerator<Context = GroupContext<C>, TypeOptionType = T>,
- {
- pub async fn new(field_rev: &Arc<FieldRevision>, mut configuration: GroupContext<C>) -> FlowyResult<Self> {
- let field_type_rev = field_rev.ty;
- let type_option = field_rev.get_type_option::<T>(field_type_rev);
- let groups = G::generate_groups(&field_rev.id, &configuration, &type_option);
- let _ = configuration.init_groups(groups, false)?;
- Ok(Self {
- field_id: field_rev.id.clone(),
- type_option,
- group_ctx: configuration,
- group_action_phantom: PhantomData,
- cell_parser_phantom: PhantomData,
- })
- }
- // https://stackoverflow.com/questions/69413164/how-to-fix-this-clippy-warning-needless-collect
- #[allow(clippy::needless_collect)]
- fn update_default_group(
- &mut self,
- row_rev: &RowRevision,
- other_group_changesets: &[GroupChangesetPB],
- ) -> GroupChangesetPB {
- let default_group = self.group_ctx.get_mut_default_group();
- // [other_group_inserted_row] contains all the inserted rows except the default group.
- let other_group_inserted_row = other_group_changesets
- .iter()
- .flat_map(|changeset| &changeset.inserted_rows)
- .collect::<Vec<&InsertedRowPB>>();
- // Calculate the inserted_rows of the default_group
- let default_group_inserted_row = other_group_changesets
- .iter()
- .flat_map(|changeset| &changeset.deleted_rows)
- .cloned()
- .filter(|row_id| {
- // if the [other_group_inserted_row] contains the row_id of the row
- // which means the row should not move to the default group.
- !other_group_inserted_row
- .iter()
- .any(|inserted_row| &inserted_row.row.id == row_id)
- })
- .collect::<Vec<String>>();
- let mut changeset = GroupChangesetPB::new(default_group.id.clone());
- if !default_group_inserted_row.is_empty() {
- changeset.inserted_rows.push(InsertedRowPB::new(row_rev.into()));
- default_group.add_row(row_rev.into());
- }
- // [other_group_delete_rows] contains all the deleted rows except the default group.
- let other_group_delete_rows: Vec<String> = other_group_changesets
- .iter()
- .flat_map(|changeset| &changeset.deleted_rows)
- .cloned()
- .collect();
- let default_group_deleted_rows = other_group_changesets
- .iter()
- .flat_map(|changeset| &changeset.inserted_rows)
- .filter(|inserted_row| {
- // if the [other_group_delete_rows] contain the inserted_row, which means this row should move
- // out from the default_group.
- let inserted_row_id = &inserted_row.row.id;
- !other_group_delete_rows.iter().any(|row_id| inserted_row_id == row_id)
- })
- .collect::<Vec<&InsertedRowPB>>();
- let mut deleted_row_ids = vec![];
- for row in &default_group.rows {
- if default_group_deleted_rows
- .iter()
- .any(|deleted_row| deleted_row.row.id == row.id)
- {
- deleted_row_ids.push(row.id.clone());
- }
- }
- default_group.rows.retain(|row| !deleted_row_ids.contains(&row.id));
- changeset.deleted_rows.extend(deleted_row_ids);
- changeset
- }
- }
- impl<C, T, G, P> GroupControllerSharedOperation for GenericGroupController<C, T, G, P>
- where
- P: CellBytesParser,
- C: GroupConfigurationContentSerde,
- T: TypeOptionDataDeserializer,
- G: GroupGenerator<Context = GroupContext<C>, TypeOptionType = T>,
- Self: GroupAction<CellDataType = P::Object>,
- {
- fn field_id(&self) -> &str {
- &self.field_id
- }
- fn groups(&self) -> Vec<Group> {
- self.group_ctx.clone_groups()
- }
- fn get_group(&self, group_id: &str) -> Option<(usize, Group)> {
- let group = self.group_ctx.get_group(group_id)?;
- Some((group.0, group.1.clone()))
- }
- #[tracing::instrument(level = "trace", skip_all, fields(row_count=%row_revs.len(), group_result))]
- fn fill_groups(&mut self, row_revs: &[Arc<RowRevision>], field_rev: &FieldRevision) -> FlowyResult<()> {
- for row_rev in row_revs {
- if let Some(cell_rev) = row_rev.cells.get(&self.field_id) {
- let mut grouped_rows: Vec<GroupedRow> = vec![];
- let cell_bytes = decode_any_cell_data(cell_rev.data.clone(), field_rev);
- let cell_data = cell_bytes.parser::<P>()?;
- for group in self.group_ctx.concrete_groups() {
- if self.can_group(&group.filter_content, &cell_data) {
- grouped_rows.push(GroupedRow {
- row: row_rev.into(),
- group_id: group.id.clone(),
- });
- }
- }
- if grouped_rows.is_empty() {
- self.group_ctx.get_mut_default_group().add_row(row_rev.into());
- } else {
- for group_row in grouped_rows {
- if let Some(group) = self.group_ctx.get_mut_group(&group_row.group_id) {
- group.add_row(group_row.row);
- }
- }
- }
- } else {
- self.group_ctx.get_mut_default_group().add_row(row_rev.into());
- }
- }
- tracing::Span::current().record("group_result", &format!("{},", self.group_ctx,).as_str());
- Ok(())
- }
- fn move_group(&mut self, from_group_id: &str, to_group_id: &str) -> FlowyResult<()> {
- self.group_ctx.move_group(from_group_id, to_group_id)
- }
- fn did_update_row(
- &mut self,
- row_rev: &RowRevision,
- field_rev: &FieldRevision,
- ) -> FlowyResult<Vec<GroupChangesetPB>> {
- if let Some(cell_rev) = row_rev.cells.get(&self.field_id) {
- let cell_bytes = decode_any_cell_data(cell_rev.data.clone(), field_rev);
- let cell_data = cell_bytes.parser::<P>()?;
- let mut changesets = self.add_row_if_match(row_rev, &cell_data);
- let default_group_changeset = self.update_default_group(row_rev, &changesets);
- tracing::info!("default_group_changeset: {}", default_group_changeset);
- if !default_group_changeset.is_empty() {
- changesets.push(default_group_changeset);
- }
- Ok(changesets)
- } else {
- Ok(vec![])
- }
- }
- fn did_delete_row(
- &mut self,
- row_rev: &RowRevision,
- field_rev: &FieldRevision,
- ) -> FlowyResult<Vec<GroupChangesetPB>> {
- if let Some(cell_rev) = row_rev.cells.get(&self.field_id) {
- let cell_bytes = decode_any_cell_data(cell_rev.data.clone(), field_rev);
- let cell_data = cell_bytes.parser::<P>()?;
- Ok(self.remove_row_if_match(row_rev, &cell_data))
- } else {
- Ok(vec![])
- }
- }
- fn move_group_row(&mut self, context: MoveGroupRowContext) -> FlowyResult<Vec<GroupChangesetPB>> {
- if let Some(cell_rev) = context.row_rev.cells.get(&self.field_id) {
- let cell_bytes = decode_any_cell_data(cell_rev.data.clone(), context.field_rev);
- let cell_data = cell_bytes.parser::<P>()?;
- Ok(self.move_row(&cell_data, context))
- } else {
- Ok(vec![])
- }
- }
- fn did_update_field(&mut self, field_rev: &FieldRevision) -> FlowyResult<Option<GroupViewChangesetPB>> {
- let type_option = field_rev.get_type_option::<T>(field_rev.ty);
- let groups = G::generate_groups(&field_rev.id, &self.group_ctx, &type_option);
- let changeset = self.group_ctx.init_groups(groups, false)?;
- Ok(changeset)
- }
- }
- struct GroupedRow {
- row: RowPB,
- group_id: String,
- }
|