use std::sync::{Arc, Weak}; use flowy_error::{FlowyError, FlowyResult}; use lib_dispatch::prelude::{data_result_ok, AFPluginData, AFPluginState, DataResult}; use crate::entities::*; use crate::manager::FolderManager; use crate::share::ImportParams; fn upgrade_folder( folder_manager: AFPluginState>, ) -> FlowyResult> { let folder = folder_manager .upgrade() .ok_or(FlowyError::internal().context("The folder manager is already dropped"))?; Ok(folder) } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn create_workspace_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let params: CreateWorkspaceParams = data.into_inner().try_into()?; let workspace = folder.create_workspace(params).await?; data_result_ok(workspace.into()) } #[tracing::instrument(level = "debug", skip(folder), err)] pub(crate) async fn get_workspace_views_handler( folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let child_views = folder.get_current_workspace_views().await?; let repeated_view: RepeatedViewPB = child_views.into(); data_result_ok(repeated_view) } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn open_workspace_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let params: WorkspaceIdPB = data.into_inner(); match params.value { None => Err(FlowyError::workspace_id().context("workspace id should not be empty")), Some(workspace_id) => { if workspace_id.is_empty() { Err(FlowyError::workspace_id().context("workspace id should not be empty")) } else { let workspace = folder.open_workspace(&workspace_id).await?; let views = folder.get_workspace_views(&workspace_id).await?; let workspace_pb: WorkspacePB = (workspace, views).into(); data_result_ok(workspace_pb) } }, } } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn read_workspaces_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let params: WorkspaceIdPB = data.into_inner(); let workspaces = match params.value { None => folder.get_all_workspaces().await, Some(workspace_id) => folder .get_workspace(&workspace_id) .await .map(|workspace| vec![workspace]) .unwrap_or_default(), }; data_result_ok(workspaces.into()) } #[tracing::instrument(level = "debug", skip(folder), err)] pub async fn get_current_workspace_setting_handler( folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let workspace = folder.get_current_workspace().await?; let latest_view: Option = folder.get_current_view().await; data_result_ok(WorkspaceSettingPB { workspace, latest_view, }) } pub(crate) async fn create_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let params: CreateViewParams = data.into_inner().try_into()?; let set_as_current = params.set_as_current; let view = folder.create_view_with_params(params).await?; if set_as_current { let _ = folder.set_current_view(&view.id).await; } data_result_ok(view_pb_without_child_views(Arc::new(view))) } pub(crate) async fn create_orphan_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let params: CreateViewParams = data.into_inner().try_into()?; let set_as_current = params.set_as_current; let view = folder.create_orphan_view_with_params(params).await?; if set_as_current { let _ = folder.set_current_view(&view.id).await; } data_result_ok(view_pb_without_child_views(Arc::new(view))) } pub(crate) async fn read_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let view_id: ViewIdPB = data.into_inner(); let view_pb = folder.get_view(&view_id.value).await?; data_result_ok(view_pb) } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn update_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let params: UpdateViewParams = data.into_inner().try_into()?; folder.update_view_with_params(params).await?; Ok(()) } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn update_view_icon_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let params: UpdateViewIconParams = data.into_inner().try_into()?; folder.update_view_icon_with_params(params).await?; Ok(()) } pub(crate) async fn delete_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let params: RepeatedViewIdPB = data.into_inner(); for view_id in ¶ms.items { let _ = folder.move_view_to_trash(view_id).await; } Ok(()) } pub(crate) async fn toggle_favorites_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let params: RepeatedViewIdPB = data.into_inner(); let folder = upgrade_folder(folder)?; for view_id in ¶ms.items { let _ = folder.toggle_favorites(view_id).await; } Ok(()) } pub(crate) async fn set_latest_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let view_id: ViewIdPB = data.into_inner(); let _ = folder.set_current_view(&view_id.value).await; Ok(()) } pub(crate) async fn close_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let view_id: ViewIdPB = data.into_inner(); let _ = folder.close_view(&view_id.value).await; Ok(()) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn move_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let params: MoveViewParams = data.into_inner().try_into()?; folder .move_view(¶ms.view_id, params.from, params.to) .await?; Ok(()) } pub(crate) async fn move_nested_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let params: MoveNestedViewParams = data.into_inner().try_into()?; folder .move_nested_view(params.view_id, params.new_parent_id, params.prev_view_id) .await?; Ok(()) } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn duplicate_view_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let view: ViewPB = data.into_inner(); folder.duplicate_view(&view.id).await?; Ok(()) } #[tracing::instrument(level = "debug", skip(folder), err)] pub(crate) async fn read_favorites_handler( folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let favorites = folder.get_all_favorites().await; let mut views = vec![]; for info in favorites { let view = folder.get_view(&info.id).await; match view { Ok(view) => { views.push(view); }, Err(err) => { return Err(err); }, } } data_result_ok(RepeatedViewPB { items: views }) } #[tracing::instrument(level = "debug", skip(folder), err)] pub(crate) async fn read_trash_handler( folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; let trash = folder.get_all_trash().await; data_result_ok(trash.into()) } #[tracing::instrument(level = "debug", skip(identifier, folder), err)] pub(crate) async fn putback_trash_handler( identifier: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; folder.restore_trash(&identifier.id).await; Ok(()) } #[tracing::instrument(level = "debug", skip(identifiers, folder), err)] pub(crate) async fn delete_trash_handler( identifiers: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let trash_ids = identifiers.into_inner().items; for trash_id in trash_ids { let _ = folder.delete_trash(&trash_id.id).await; } Ok(()) } #[tracing::instrument(level = "debug", skip(folder), err)] pub(crate) async fn restore_all_trash_handler( folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; folder.restore_all_trash().await; Ok(()) } #[tracing::instrument(level = "debug", skip(folder), err)] pub(crate) async fn delete_all_trash_handler( folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; folder.delete_all_trash().await; Ok(()) } #[tracing::instrument(level = "debug", skip(data, folder), err)] pub(crate) async fn import_data_handler( data: AFPluginData, folder: AFPluginState>, ) -> Result<(), FlowyError> { let folder = upgrade_folder(folder)?; let params: ImportParams = data.into_inner().try_into()?; folder.import(params).await?; Ok(()) } #[tracing::instrument(level = "debug", skip(folder), err)] pub(crate) async fn get_folder_snapshots_handler( data: AFPluginData, folder: AFPluginState>, ) -> DataResult { let folder = upgrade_folder(folder)?; if let Some(workspace_id) = &data.value { let snapshots = folder.get_folder_snapshots(workspace_id, 10).await?; data_result_ok(RepeatedFolderSnapshotPB { items: snapshots }) } else { data_result_ok(RepeatedFolderSnapshotPB { items: vec![] }) } }