12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061 |
- use std::collections::HashMap;
- use std::ops::Deref;
- use std::sync::Arc;
- use bytes::Bytes;
- use collab_database::database::{gen_row_id, timestamp, Database as InnerDatabase};
- use collab_database::fields::{Field, TypeOptionData};
- use collab_database::rows::{Cell, Cells, Row, RowCell, RowId};
- use collab_database::views::{DatabaseLayout, DatabaseView, LayoutSetting};
- use parking_lot::Mutex;
- use tokio::sync::{broadcast, RwLock};
- use flowy_error::{internal_error, FlowyError, FlowyResult};
- use flowy_task::TaskDispatcher;
- use lib_infra::future::{to_fut, Fut};
- use crate::entities::{
- AlterFilterParams, AlterSortParams, CalendarEventPB, CellChangesetNotifyPB, CellPB,
- CreateRowParams, DatabaseFieldChangesetPB, DatabasePB, DatabaseViewSettingPB, DeleteFilterParams,
- DeleteGroupParams, DeleteSortParams, FieldChangesetParams, FieldIdPB, FieldPB, FieldType,
- GroupPB, IndexFieldPB, InsertGroupParams, InsertedRowPB, LayoutSettingParams, RepeatedFilterPB,
- RepeatedGroupPB, RepeatedSortPB, RowPB, RowsChangesetPB, SelectOptionCellDataPB, SelectOptionPB,
- };
- use crate::notification::{send_notification, DatabaseNotification};
- use crate::services::cell::{
- apply_cell_changeset, get_cell_protobuf, insert_date_cell, AnyTypeCache, CellBuilder, CellCache,
- ToCellChangeset,
- };
- use crate::services::database::util::database_view_setting_pb_from_view;
- use crate::services::database_view::{DatabaseViewChanged, DatabaseViewData, DatabaseViews};
- use crate::services::field::{
- default_type_option_data_from_type, select_type_option_from_field, transform_type_option,
- type_option_data_from_pb_or_default, type_option_to_pb, SelectOptionCellChangeset,
- SelectOptionIds, TypeOptionCellDataHandler, TypeOptionCellExt,
- };
- use crate::services::filter::Filter;
- use crate::services::group::{default_group_setting, GroupSetting, RowChangeset};
- use crate::services::share::csv::{CSVExport, ExportStyle};
- use crate::services::sort::Sort;
- #[derive(Clone)]
- pub struct DatabaseEditor {
- database: MutexDatabase,
- pub cell_cache: CellCache,
- database_views: Arc<DatabaseViews>,
- }
- impl DatabaseEditor {
- pub async fn new(
- database: MutexDatabase,
- task_scheduler: Arc<RwLock<TaskDispatcher>>,
- ) -> FlowyResult<Self> {
- let cell_cache = AnyTypeCache::<u64>::new();
- let database_view_data = Arc::new(DatabaseViewDataImpl {
- database: database.clone(),
- task_scheduler: task_scheduler.clone(),
- cell_cache: cell_cache.clone(),
- });
- let database_views =
- Arc::new(DatabaseViews::new(database.clone(), cell_cache.clone(), database_view_data).await?);
- Ok(Self {
- database,
- cell_cache,
- database_views,
- })
- }
- #[tracing::instrument(level = "debug", skip_all)]
- pub async fn close_view_editor(&self, view_id: &str) -> bool {
- self.database_views.close_view(view_id).await
- }
- pub async fn close(&self) {}
- pub async fn subscribe_view_changed(
- &self,
- view_id: &str,
- ) -> FlowyResult<broadcast::Receiver<DatabaseViewChanged>> {
- let view_editor = self.database_views.get_view_editor(view_id).await?;
- Ok(view_editor.notifier.subscribe())
- }
- pub fn get_field(&self, field_id: &str) -> Option<Field> {
- self.database.lock().fields.get_field(field_id)
- }
- pub async fn insert_group(&self, params: InsertGroupParams) -> FlowyResult<()> {
- {
- let database = self.database.lock();
- let field = database.fields.get_field(¶ms.field_id);
- if let Some(field) = field {
- let group_setting = default_group_setting(&field);
- database.insert_group_setting(¶ms.view_id, group_setting);
- }
- }
- let view_editor = self.database_views.get_view_editor(¶ms.view_id).await?;
- view_editor.v_initialize_new_group(params).await?;
- Ok(())
- }
- pub async fn delete_group(&self, params: DeleteGroupParams) -> FlowyResult<()> {
- self
- .database
- .lock()
- .delete_group_setting(¶ms.view_id, ¶ms.group_id);
- let view_editor = self.database_views.get_view_editor(¶ms.view_id).await?;
- view_editor.v_delete_group(params).await?;
- Ok(())
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- pub async fn create_or_update_filter(&self, params: AlterFilterParams) -> FlowyResult<()> {
- let view_editor = self.database_views.get_view_editor(¶ms.view_id).await?;
- view_editor.v_insert_filter(params).await?;
- Ok(())
- }
- pub async fn delete_filter(&self, params: DeleteFilterParams) -> FlowyResult<()> {
- let view_editor = self.database_views.get_view_editor(¶ms.view_id).await?;
- view_editor.v_delete_filter(params).await?;
- Ok(())
- }
- pub async fn create_or_update_sort(&self, params: AlterSortParams) -> FlowyResult<Sort> {
- let view_editor = self.database_views.get_view_editor(¶ms.view_id).await?;
- let sort = view_editor.v_insert_sort(params).await?;
- Ok(sort)
- }
- pub async fn delete_sort(&self, params: DeleteSortParams) -> FlowyResult<()> {
- let view_editor = self.database_views.get_view_editor(¶ms.view_id).await?;
- view_editor.v_delete_sort(params).await?;
- Ok(())
- }
- pub async fn get_all_filters(&self, view_id: &str) -> RepeatedFilterPB {
- if let Ok(view_editor) = self.database_views.get_view_editor(view_id).await {
- view_editor.v_get_all_filters().await.into()
- } else {
- RepeatedFilterPB { items: vec![] }
- }
- }
- pub async fn get_filter(&self, view_id: &str, filter_id: &str) -> Option<Filter> {
- if let Ok(view_editor) = self.database_views.get_view_editor(view_id).await {
- Some(view_editor.v_get_filter(filter_id).await?)
- } else {
- None
- }
- }
- pub async fn get_all_sorts(&self, view_id: &str) -> RepeatedSortPB {
- if let Ok(view_editor) = self.database_views.get_view_editor(view_id).await {
- view_editor.v_get_all_sorts().await.into()
- } else {
- RepeatedSortPB { items: vec![] }
- }
- }
- pub async fn delete_all_sorts(&self, view_id: &str) {
- if let Ok(view_editor) = self.database_views.get_view_editor(view_id).await {
- let _ = view_editor.v_delete_all_sorts().await;
- }
- }
- pub fn get_fields(&self, view_id: &str, field_ids: Option<Vec<String>>) -> Vec<Field> {
- self.database.lock().get_fields(view_id, field_ids)
- }
- pub async fn update_field(&self, params: FieldChangesetParams) -> FlowyResult<()> {
- self
- .database
- .lock()
- .fields
- .update_field(¶ms.field_id, |update| {
- update
- .set_name_if_not_none(params.name)
- .set_field_type_if_not_none(params.field_type.map(|field_type| field_type.into()))
- .set_width_at_if_not_none(params.width.map(|value| value as i64))
- .set_visibility_if_not_none(params.visibility);
- });
- self
- .notify_did_update_database_field(¶ms.field_id)
- .await?;
- Ok(())
- }
- pub async fn delete_field(&self, field_id: &str) -> FlowyResult<()> {
- let database_id = {
- let database = self.database.lock();
- database.delete_field(field_id);
- database.get_database_id()
- };
- let notified_changeset =
- DatabaseFieldChangesetPB::delete(&database_id, vec![FieldIdPB::from(field_id)]);
- self.notify_did_update_database(notified_changeset).await?;
- Ok(())
- }
- pub async fn update_field_type_option(
- &self,
- view_id: &str,
- field_id: &str,
- type_option_data: TypeOptionData,
- old_field: Field,
- ) -> FlowyResult<()> {
- let field_type = FieldType::from(old_field.field_type);
- self
- .database
- .lock()
- .fields
- .update_field(field_id, |update| {
- update.update_type_options(|type_options_update| {
- type_options_update.insert(&field_type.to_string(), type_option_data);
- });
- });
- self
- .database_views
- .did_update_field_type_option(view_id, field_id, &old_field)
- .await?;
- let _ = self.notify_did_update_database_field(field_id).await;
- Ok(())
- }
- pub async fn switch_to_field_type(
- &self,
- field_id: &str,
- new_field_type: &FieldType,
- ) -> FlowyResult<()> {
- let field = self.database.lock().fields.get_field(field_id);
- match field {
- None => {},
- Some(field) => {
- let old_field_type = FieldType::from(field.field_type);
- let old_type_option = field.get_any_type_option(old_field_type.clone());
- let new_type_option = field
- .get_any_type_option(new_field_type)
- .unwrap_or_else(|| default_type_option_data_from_type(new_field_type));
- let transformed_type_option = transform_type_option(
- &new_type_option,
- new_field_type,
- old_type_option,
- old_field_type,
- );
- self
- .database
- .lock()
- .fields
- .update_field(field_id, |update| {
- update
- .set_field_type(new_field_type.into())
- .set_type_option(new_field_type.into(), Some(transformed_type_option));
- });
- },
- }
- self.notify_did_update_database_field(field_id).await?;
- Ok(())
- }
- pub async fn duplicate_field(&self, view_id: &str, field_id: &str) -> FlowyResult<()> {
- let value = self
- .database
- .lock()
- .duplicate_field(view_id, field_id, |field| format!("{} (copy)", field.name));
- if let Some((index, duplicated_field)) = value {
- let _ = self
- .notify_did_insert_database_field(duplicated_field, index)
- .await;
- }
- Ok(())
- }
- pub async fn duplicate_row(&self, view_id: &str, row_id: &RowId) {
- let _ = self.database.lock().duplicate_row(view_id, row_id);
- }
- pub async fn move_row(&self, view_id: &str, from: RowId, to: RowId) {
- let database = self.database.lock();
- if let (Some(row), Some(from_index), Some(to_index)) = (
- database.get_row(&from),
- database.index_of_row(view_id, &from),
- database.index_of_row(view_id, &to),
- ) {
- database.views.update_database_view(view_id, |view| {
- view.move_row_order(from_index as u32, to_index as u32);
- });
- drop(database);
- let delete_row_id = from.into_inner();
- let insert_row = InsertedRowPB::from(&row).with_index(to_index as i32);
- let changeset =
- RowsChangesetPB::from_move(view_id.to_string(), vec![delete_row_id], vec![insert_row]);
- send_notification(view_id, DatabaseNotification::DidUpdateViewRows)
- .payload(changeset)
- .send();
- }
- }
- pub async fn create_row(&self, params: CreateRowParams) -> FlowyResult<Option<Row>> {
- let fields = self.database.lock().get_fields(¶ms.view_id, None);
- let mut cells =
- CellBuilder::with_cells(params.cell_data_by_field_id.unwrap_or_default(), &fields).build();
- for view in self.database_views.editors().await {
- view.v_will_create_row(&mut cells, ¶ms.group_id).await;
- }
- let result = self.database.lock().create_row_in_view(
- ¶ms.view_id,
- collab_database::rows::CreateRowParams {
- id: gen_row_id(),
- cells,
- height: 60,
- visibility: true,
- prev_row_id: params.start_row_id,
- timestamp: timestamp(),
- },
- );
- if let Some((index, row_order)) = result {
- let row = self.database.lock().get_row(&row_order.id);
- if let Some(row) = row {
- for view in self.database_views.editors().await {
- view.v_did_create_row(&row, ¶ms.group_id, index).await;
- }
- return Ok(Some(row));
- }
- }
- Ok(None)
- }
- pub async fn get_field_type_option_data(&self, field_id: &str) -> Option<(Field, Bytes)> {
- let field = self.database.lock().fields.get_field(field_id);
- field.map(|field| {
- let field_type = FieldType::from(field.field_type);
- let type_option = field
- .get_any_type_option(field_type.clone())
- .unwrap_or_else(|| default_type_option_data_from_type(&field_type));
- (field, type_option_to_pb(type_option, &field_type))
- })
- }
- pub async fn create_field_with_type_option(
- &self,
- view_id: &str,
- field_type: &FieldType,
- type_option_data: Option<Vec<u8>>,
- ) -> (Field, Bytes) {
- let name = field_type.default_name();
- let type_option_data = match type_option_data {
- None => default_type_option_data_from_type(field_type),
- Some(type_option_data) => type_option_data_from_pb_or_default(type_option_data, field_type),
- };
- let (index, field) =
- self
- .database
- .lock()
- .create_default_field(view_id, name, field_type.into(), |field| {
- field
- .type_options
- .insert(field_type.to_string(), type_option_data.clone());
- });
- let _ = self
- .notify_did_insert_database_field(field.clone(), index)
- .await;
- (field, type_option_to_pb(type_option_data, field_type))
- }
- pub async fn move_field(
- &self,
- view_id: &str,
- field_id: &str,
- from: i32,
- to: i32,
- ) -> FlowyResult<()> {
- let (database_id, field) = {
- let database = self.database.lock();
- database.views.update_database_view(view_id, |view_update| {
- view_update.move_field_order(from as u32, to as u32);
- });
- let field = database.fields.get_field(field_id);
- let database_id = database.get_database_id();
- (database_id, field)
- };
- if let Some(field) = field {
- let delete_field = FieldIdPB::from(field_id);
- let insert_field = IndexFieldPB::from_field(field, to as usize);
- let notified_changeset = DatabaseFieldChangesetPB {
- view_id: database_id,
- inserted_fields: vec![insert_field],
- deleted_fields: vec![delete_field],
- updated_fields: vec![],
- };
- self.notify_did_update_database(notified_changeset).await?;
- }
- Ok(())
- }
- pub async fn get_rows(&self, view_id: &str) -> FlowyResult<Vec<Arc<Row>>> {
- let view_editor = self.database_views.get_view_editor(view_id).await?;
- Ok(view_editor.v_get_rows().await)
- }
- pub fn get_row(&self, row_id: &RowId) -> Option<Row> {
- self.database.lock().get_row(row_id)
- }
- pub async fn delete_row(&self, row_id: &RowId) {
- let row = self.database.lock().remove_row(row_id);
- if let Some(row) = row {
- tracing::trace!("Did delete row:{:?}", row);
- for view in self.database_views.editors().await {
- view.v_did_delete_row(&row).await;
- }
- }
- }
- pub async fn get_cell(&self, field_id: &str, row_id: RowId) -> CellPB {
- let (field, cell) = {
- let database = self.database.lock();
- let field = database.fields.get_field(field_id);
- let cell = database.get_cell(field_id, &row_id);
- (field, cell)
- };
- match (field, cell) {
- (Some(field), Some(cell)) => {
- let field_type = FieldType::from(field.field_type);
- let cell_bytes = get_cell_protobuf(&cell, &field, Some(self.cell_cache.clone()));
- CellPB {
- field_id: field_id.to_string(),
- row_id: row_id.into(),
- data: cell_bytes.to_vec(),
- field_type: Some(field_type),
- }
- },
- _ => CellPB::empty(field_id, row_id.into_inner()),
- }
- }
- pub async fn get_cells_for_field(&self, view_id: &str, field_id: &str) -> Vec<RowCell> {
- self.database.lock().get_cells_for_field(view_id, field_id)
- }
- pub async fn update_cell_with_changeset<T>(
- &self,
- view_id: &str,
- row_id: RowId,
- field_id: &str,
- cell_changeset: T,
- ) -> FlowyResult<()>
- where
- T: ToCellChangeset,
- {
- let (field, cell) = {
- let database = self.database.lock();
- let field = match database.fields.get_field(field_id) {
- Some(field) => Ok(field),
- None => {
- let msg = format!("Field with id:{} not found", &field_id);
- Err(FlowyError::internal().context(msg))
- },
- }?;
- (
- field,
- database
- .get_cell(field_id, &row_id)
- .map(|row_cell| row_cell.cell),
- )
- };
- let new_cell =
- apply_cell_changeset(cell_changeset, cell, &field, Some(self.cell_cache.clone()))?;
- self.update_cell(view_id, row_id, field_id, new_cell).await
- }
- /// Update a cell in the database.
- /// This will notify all views that the cell has been updated.
- pub async fn update_cell(
- &self,
- view_id: &str,
- row_id: RowId,
- field_id: &str,
- new_cell: Cell,
- ) -> FlowyResult<()> {
- // Get the old row before updating the cell. It would be better to get the old cell
- let old_row = { self.database.lock().get_row(&row_id) };
- // Get all the updated_at fields. We will update all of them.
- let updated_at_fields = self
- .database
- .lock()
- .get_fields(view_id, None)
- .into_iter()
- .filter(|f| FieldType::from(f.field_type) == FieldType::UpdatedAt)
- .collect::<Vec<Field>>();
- self.database.lock().update_row(&row_id, |row_update| {
- row_update.update_cells(|cell_update| {
- let mut cells_update = cell_update.insert(field_id, new_cell);
- for field in &updated_at_fields {
- cells_update =
- cells_update.insert(&field.id, insert_date_cell(timestamp(), Some(true), field));
- }
- });
- });
- let option_row = self.database.lock().get_row(&row_id);
- if let Some(new_row) = option_row {
- for view in self.database_views.editors().await {
- view.v_did_update_row(&old_row, &new_row, field_id).await;
- }
- }
- // Collect all the updated field's id. Notify the frontend that all of them have been updated.
- let mut updated_field_ids = updated_at_fields
- .into_iter()
- .map(|field| field.id)
- .collect::<Vec<String>>();
- updated_field_ids.push(field_id.to_string());
- let changeset = updated_field_ids
- .into_iter()
- .map(|field_id| CellChangesetNotifyPB {
- view_id: view_id.to_string(),
- row_id: row_id.clone().into_inner(),
- field_id,
- })
- .collect();
- notify_did_update_cell(changeset).await;
- Ok(())
- }
- pub async fn create_select_option(
- &self,
- field_id: &str,
- option_name: String,
- ) -> Option<SelectOptionPB> {
- let field = self.database.lock().fields.get_field(field_id)?;
- let type_option = select_type_option_from_field(&field).ok()?;
- let select_option = type_option.create_option(&option_name);
- Some(SelectOptionPB::from(select_option))
- }
- pub async fn insert_select_options(
- &self,
- view_id: &str,
- field_id: &str,
- row_id: RowId,
- options: Vec<SelectOptionPB>,
- ) -> FlowyResult<()> {
- let field = match self.database.lock().fields.get_field(field_id) {
- Some(field) => Ok(field),
- None => {
- let msg = format!("Field with id:{} not found", &field_id);
- Err(FlowyError::internal().context(msg))
- },
- }?;
- let mut type_option = select_type_option_from_field(&field)?;
- let cell_changeset = SelectOptionCellChangeset {
- insert_option_ids: options.iter().map(|option| option.id.clone()).collect(),
- ..Default::default()
- };
- for option in options {
- type_option.insert_option(option.into());
- }
- self
- .database
- .lock()
- .fields
- .update_field(field_id, |update| {
- update.set_type_option(field.field_type, Some(type_option.to_type_option_data()));
- });
- self
- .update_cell_with_changeset(view_id, row_id, field_id, cell_changeset)
- .await?;
- Ok(())
- }
- pub async fn delete_select_options(
- &self,
- view_id: &str,
- field_id: &str,
- row_id: RowId,
- options: Vec<SelectOptionPB>,
- ) -> FlowyResult<()> {
- let field = match self.database.lock().fields.get_field(field_id) {
- Some(field) => Ok(field),
- None => {
- let msg = format!("Field with id:{} not found", &field_id);
- Err(FlowyError::internal().context(msg))
- },
- }?;
- let mut type_option = select_type_option_from_field(&field)?;
- let cell_changeset = SelectOptionCellChangeset {
- delete_option_ids: options.iter().map(|option| option.id.clone()).collect(),
- ..Default::default()
- };
- for option in options {
- type_option.delete_option(option.into());
- }
- self
- .database
- .lock()
- .fields
- .update_field(field_id, |update| {
- update.set_type_option(field.field_type, Some(type_option.to_type_option_data()));
- });
- self
- .update_cell_with_changeset(view_id, row_id, field_id, cell_changeset)
- .await?;
- Ok(())
- }
- pub async fn get_select_options(&self, row_id: RowId, field_id: &str) -> SelectOptionCellDataPB {
- let field = self.database.lock().fields.get_field(field_id);
- match field {
- None => SelectOptionCellDataPB::default(),
- Some(field) => {
- let row_cell = self.database.lock().get_cell(field_id, &row_id);
- let ids = match row_cell {
- None => SelectOptionIds::new(),
- Some(row_cell) => SelectOptionIds::from(&row_cell.cell),
- };
- match select_type_option_from_field(&field) {
- Ok(type_option) => type_option.get_selected_options(ids).into(),
- Err(_) => SelectOptionCellDataPB::default(),
- }
- },
- }
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- pub async fn load_groups(&self, view_id: &str) -> FlowyResult<RepeatedGroupPB> {
- let view = self.database_views.get_view_editor(view_id).await?;
- let groups = view.v_load_groups().await?;
- Ok(RepeatedGroupPB { items: groups })
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- pub async fn get_group(&self, view_id: &str, group_id: &str) -> FlowyResult<GroupPB> {
- let view = self.database_views.get_view_editor(view_id).await?;
- let group = view.v_get_group(group_id).await?;
- Ok(group)
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- pub async fn move_group(
- &self,
- view_id: &str,
- from_group: &str,
- to_group: &str,
- ) -> FlowyResult<()> {
- let view = self.database_views.get_view_editor(view_id).await?;
- view.v_move_group(from_group, to_group).await?;
- Ok(())
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- pub async fn move_group_row(
- &self,
- view_id: &str,
- to_group: &str,
- from_row: RowId,
- to_row: Option<RowId>,
- ) -> FlowyResult<()> {
- let row = self.database.lock().get_row(&from_row);
- match row {
- None => {
- tracing::warn!(
- "Move row between group failed, can not find the row:{}",
- from_row
- )
- },
- Some(row) => {
- let mut row_changeset = RowChangeset::new(row.id.clone());
- let view = self.database_views.get_view_editor(view_id).await?;
- view
- .v_move_group_row(&row, &mut row_changeset, to_group, to_row)
- .await;
- tracing::trace!("Row data changed: {:?}", row_changeset);
- self.database.lock().update_row(&row.id, |row| {
- row.set_cells(Cells::from(row_changeset.cell_by_field_id.clone()));
- });
- let cell_changesets = cell_changesets_from_cell_by_field_id(
- view_id,
- row_changeset.row_id,
- row_changeset.cell_by_field_id,
- );
- notify_did_update_cell(cell_changesets).await;
- },
- }
- Ok(())
- }
- pub async fn group_by_field(&self, view_id: &str, field_id: &str) -> FlowyResult<()> {
- let view = self.database_views.get_view_editor(view_id).await?;
- view.v_update_group_setting(field_id).await?;
- Ok(())
- }
- pub async fn set_layout_setting(
- &self,
- view_id: &str,
- layout_ty: DatabaseLayout,
- layout_setting: LayoutSettingParams,
- ) {
- if let Ok(view) = self.database_views.get_view_editor(view_id).await {
- let _ = view.v_set_layout_settings(&layout_ty, layout_setting).await;
- }
- }
- pub async fn get_layout_setting(
- &self,
- view_id: &str,
- layout_ty: DatabaseLayout,
- ) -> Option<LayoutSettingParams> {
- let view = self.database_views.get_view_editor(view_id).await.ok()?;
- let layout_setting = view.v_get_layout_settings(&layout_ty).await;
- Some(layout_setting)
- }
- #[tracing::instrument(level = "trace", skip_all)]
- pub async fn get_all_calendar_events(&self, view_id: &str) -> Vec<CalendarEventPB> {
- match self.database_views.get_view_editor(view_id).await {
- Ok(view) => view.v_get_all_calendar_events().await.unwrap_or_default(),
- Err(_) => {
- tracing::warn!("Can not find the view: {}", view_id);
- vec![]
- },
- }
- }
- #[tracing::instrument(level = "trace", skip_all)]
- pub async fn get_calendar_event(&self, view_id: &str, row_id: RowId) -> Option<CalendarEventPB> {
- let view = self.database_views.get_view_editor(view_id).await.ok()?;
- view.v_get_calendar_event(row_id).await
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- async fn notify_did_insert_database_field(&self, field: Field, index: usize) -> FlowyResult<()> {
- let database_id = self.database.lock().get_database_id();
- let index_field = IndexFieldPB::from_field(field, index);
- let notified_changeset = DatabaseFieldChangesetPB::insert(&database_id, vec![index_field]);
- let _ = self.notify_did_update_database(notified_changeset).await;
- Ok(())
- }
- #[tracing::instrument(level = "trace", skip_all, err)]
- async fn notify_did_update_database_field(&self, field_id: &str) -> FlowyResult<()> {
- let (database_id, field) = {
- let database = self.database.lock();
- let database_id = database.get_database_id();
- let field = database.fields.get_field(field_id);
- (database_id, field)
- };
- if let Some(field) = field {
- let updated_field = FieldPB::from(field);
- let notified_changeset =
- DatabaseFieldChangesetPB::update(&database_id, vec![updated_field.clone()]);
- self.notify_did_update_database(notified_changeset).await?;
- send_notification(field_id, DatabaseNotification::DidUpdateField)
- .payload(updated_field)
- .send();
- }
- Ok(())
- }
- async fn notify_did_update_database(
- &self,
- changeset: DatabaseFieldChangesetPB,
- ) -> FlowyResult<()> {
- let views = self.database.lock().get_all_views_description();
- for view in views {
- send_notification(&view.id, DatabaseNotification::DidUpdateFields)
- .payload(changeset.clone())
- .send();
- }
- Ok(())
- }
- pub async fn get_database_view_setting(
- &self,
- view_id: &str,
- ) -> FlowyResult<DatabaseViewSettingPB> {
- let view = self
- .database
- .lock()
- .get_view(view_id)
- .ok_or_else(|| FlowyError::record_not_found().context("Can't find the database view"))?;
- Ok(database_view_setting_pb_from_view(view))
- }
- pub async fn get_database_data(&self, view_id: &str) -> DatabasePB {
- let rows = self.get_rows(view_id).await.unwrap_or_default();
- let (database_id, fields) = {
- let database = self.database.lock();
- let database_id = database.get_database_id();
- let fields = database
- .fields
- .get_all_field_orders()
- .into_iter()
- .map(FieldIdPB::from)
- .collect();
- (database_id, fields)
- };
- let rows = rows
- .into_iter()
- .map(|row| RowPB::from(row.as_ref()))
- .collect::<Vec<RowPB>>();
- DatabasePB {
- id: database_id,
- fields,
- rows,
- }
- }
- pub async fn export_csv(&self, style: ExportStyle) -> FlowyResult<String> {
- let database = self.database.clone();
- let csv = tokio::task::spawn_blocking(move || {
- let database_guard = database.lock();
- let csv = CSVExport.export_database(&database_guard, style)?;
- Ok::<String, FlowyError>(csv)
- })
- .await
- .map_err(internal_error)??;
- Ok(csv)
- }
- }
- pub(crate) async fn notify_did_update_cell(changesets: Vec<CellChangesetNotifyPB>) {
- for changeset in changesets {
- let id = format!("{}:{}", changeset.row_id, changeset.field_id);
- send_notification(&id, DatabaseNotification::DidUpdateCell).send();
- }
- }
- fn cell_changesets_from_cell_by_field_id(
- view_id: &str,
- row_id: RowId,
- cell_by_field_id: HashMap<String, Cell>,
- ) -> Vec<CellChangesetNotifyPB> {
- let row_id = row_id.into_inner();
- cell_by_field_id
- .into_iter()
- .map(|(field_id, _cell)| CellChangesetNotifyPB {
- view_id: view_id.to_string(),
- row_id: row_id.clone(),
- field_id,
- })
- .collect()
- }
- #[derive(Clone)]
- pub struct MutexDatabase(Arc<Mutex<Arc<InnerDatabase>>>);
- impl MutexDatabase {
- pub(crate) fn new(database: Arc<InnerDatabase>) -> Self {
- Self(Arc::new(Mutex::new(database)))
- }
- }
- impl Deref for MutexDatabase {
- type Target = Arc<Mutex<Arc<InnerDatabase>>>;
- fn deref(&self) -> &Self::Target {
- &self.0
- }
- }
- unsafe impl Sync for MutexDatabase {}
- unsafe impl Send for MutexDatabase {}
- struct DatabaseViewDataImpl {
- database: MutexDatabase,
- task_scheduler: Arc<RwLock<TaskDispatcher>>,
- cell_cache: CellCache,
- }
- impl DatabaseViewData for DatabaseViewDataImpl {
- fn get_view_setting(&self, view_id: &str) -> Fut<Option<DatabaseView>> {
- let view = self.database.lock().get_view(view_id);
- to_fut(async move { view })
- }
- fn get_fields(&self, view_id: &str, field_ids: Option<Vec<String>>) -> Fut<Vec<Arc<Field>>> {
- let fields = self.database.lock().get_fields(view_id, field_ids);
- to_fut(async move { fields.into_iter().map(Arc::new).collect() })
- }
- fn get_field(&self, field_id: &str) -> Fut<Option<Arc<Field>>> {
- let field = self
- .database
- .lock()
- .fields
- .get_field(field_id)
- .map(Arc::new);
- to_fut(async move { field })
- }
- fn get_primary_field(&self) -> Fut<Option<Arc<Field>>> {
- let field = self
- .database
- .lock()
- .fields
- .get_primary_field()
- .map(Arc::new);
- to_fut(async move { field })
- }
- fn index_of_row(&self, view_id: &str, row_id: &RowId) -> Fut<Option<usize>> {
- let index = self.database.lock().index_of_row(view_id, row_id);
- to_fut(async move { index })
- }
- fn get_row(&self, view_id: &str, row_id: &RowId) -> Fut<Option<(usize, Arc<Row>)>> {
- let index = self.database.lock().index_of_row(view_id, row_id);
- let row = self.database.lock().get_row(row_id);
- to_fut(async move {
- match (index, row) {
- (Some(index), Some(row)) => Some((index, Arc::new(row))),
- _ => None,
- }
- })
- }
- fn get_rows(&self, view_id: &str) -> Fut<Vec<Arc<Row>>> {
- let rows = self.database.lock().get_rows_for_view(view_id);
- to_fut(async move { rows.into_iter().map(Arc::new).collect() })
- }
- fn get_cells_for_field(&self, view_id: &str, field_id: &str) -> Fut<Vec<Arc<RowCell>>> {
- let cells = self.database.lock().get_cells_for_field(view_id, field_id);
- to_fut(async move { cells.into_iter().map(Arc::new).collect() })
- }
- fn get_cell_in_row(&self, field_id: &str, row_id: &RowId) -> Fut<Option<Arc<RowCell>>> {
- let cell = self.database.lock().get_cell(field_id, row_id);
- to_fut(async move { cell.map(Arc::new) })
- }
- fn get_layout_for_view(&self, view_id: &str) -> DatabaseLayout {
- self
- .database
- .lock()
- .views
- .get_view_layout(view_id)
- .unwrap_or_default()
- }
- fn get_group_setting(&self, view_id: &str) -> Vec<GroupSetting> {
- self.database.lock().get_all_group_setting(view_id)
- }
- fn insert_group_setting(&self, view_id: &str, setting: GroupSetting) {
- self.database.lock().insert_group_setting(view_id, setting);
- }
- fn get_sort(&self, view_id: &str, sort_id: &str) -> Option<Sort> {
- self.database.lock().get_sort::<Sort>(view_id, sort_id)
- }
- fn insert_sort(&self, view_id: &str, sort: Sort) {
- self.database.lock().insert_sort(view_id, sort);
- }
- fn remove_sort(&self, view_id: &str, sort_id: &str) {
- self.database.lock().remove_sort(view_id, sort_id);
- }
- fn get_all_sorts(&self, view_id: &str) -> Vec<Sort> {
- self.database.lock().get_all_sorts::<Sort>(view_id)
- }
- fn remove_all_sorts(&self, view_id: &str) {
- self.database.lock().remove_all_sorts(view_id);
- }
- fn get_all_filters(&self, view_id: &str) -> Vec<Arc<Filter>> {
- self
- .database
- .lock()
- .get_all_filters(view_id)
- .into_iter()
- .map(Arc::new)
- .collect()
- }
- fn delete_filter(&self, view_id: &str, filter_id: &str) {
- self.database.lock().remove_filter(view_id, filter_id);
- }
- fn insert_filter(&self, view_id: &str, filter: Filter) {
- self.database.lock().insert_filter(view_id, filter);
- }
- fn get_filter(&self, view_id: &str, filter_id: &str) -> Option<Filter> {
- self
- .database
- .lock()
- .get_filter::<Filter>(view_id, filter_id)
- }
- fn get_filter_by_field_id(&self, view_id: &str, field_id: &str) -> Option<Filter> {
- self
- .database
- .lock()
- .get_filter_by_field_id::<Filter>(view_id, field_id)
- }
- fn get_layout_setting(&self, view_id: &str, layout_ty: &DatabaseLayout) -> Option<LayoutSetting> {
- self
- .database
- .lock()
- .views
- .get_layout_setting(view_id, layout_ty)
- }
- fn insert_layout_setting(
- &self,
- view_id: &str,
- layout_ty: &DatabaseLayout,
- layout_setting: LayoutSetting,
- ) {
- self
- .database
- .lock()
- .insert_layout_setting(view_id, layout_ty, layout_setting);
- }
- fn get_task_scheduler(&self) -> Arc<RwLock<TaskDispatcher>> {
- self.task_scheduler.clone()
- }
- fn get_type_option_cell_handler(
- &self,
- field: &Field,
- field_type: &FieldType,
- ) -> Option<Box<dyn TypeOptionCellDataHandler>> {
- TypeOptionCellExt::new_with_cell_data_cache(field, Some(self.cell_cache.clone()))
- .get_type_option_cell_data_handler(field_type)
- }
- }
|