mirror of
https://github.com/AppFlowy-IO/AppFlowy.git
synced 2024-08-30 18:12:39 +00:00
add flowy-editor crate
This commit is contained in:
parent
a6033e3359
commit
fa061ea832
@ -13,6 +13,7 @@ members = [
|
|||||||
"flowy-infra",
|
"flowy-infra",
|
||||||
"flowy-workspace",
|
"flowy-workspace",
|
||||||
"flowy-observable",
|
"flowy-observable",
|
||||||
|
"flowy-editor",
|
||||||
]
|
]
|
||||||
|
|
||||||
[profile.dev]
|
[profile.dev]
|
||||||
|
15
rust-lib/flowy-editor/Cargo.toml
Normal file
15
rust-lib/flowy-editor/Cargo.toml
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
[package]
|
||||||
|
name = "flowy-editor"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2018"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
derive_more = {version = "0.99", features = ["display"]}
|
||||||
|
flowy-dispatch = { path = "../flowy-dispatch" }
|
||||||
|
flowy-log = { path = "../flowy-log" }
|
||||||
|
flowy-derive = { path = "../flowy-derive" }
|
||||||
|
flowy-database = { path = "../flowy-database" }
|
||||||
|
diesel = {version = "1.4.7", features = ["sqlite"]}
|
||||||
|
diesel_derives = {version = "1.4.1", features = ["sqlite"]}
|
3
rust-lib/flowy-editor/Flowy.toml
Normal file
3
rust-lib/flowy-editor/Flowy.toml
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
|
||||||
|
proto_crates = ["src/entities", "src/event.rs", "src/errors.rs"]
|
||||||
|
event_files = ["src/event.rs"]
|
44
rust-lib/flowy-editor/src/entities/doc/doc_create.rs
Normal file
44
rust-lib/flowy-editor/src/entities/doc/doc_create.rs
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
use crate::{
|
||||||
|
entities::doc::parser::*,
|
||||||
|
errors::{ErrorBuilder, *},
|
||||||
|
};
|
||||||
|
use flowy_derive::ProtoBuf;
|
||||||
|
use std::convert::TryInto;
|
||||||
|
|
||||||
|
#[derive(ProtoBuf, Default)]
|
||||||
|
pub struct CreateDocRequest {
|
||||||
|
#[pb(index = 1)]
|
||||||
|
view_id: String,
|
||||||
|
|
||||||
|
#[pb(index = 2)]
|
||||||
|
pub name: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct CreateDocParams {
|
||||||
|
pub view_id: String,
|
||||||
|
pub name: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryInto<CreateDocParams> for CreateDocRequest {
|
||||||
|
type Error = WorkspaceError;
|
||||||
|
|
||||||
|
fn try_into(self) -> Result<CreateDocParams, Self::Error> {
|
||||||
|
let name = DocName::parse(self.name)
|
||||||
|
.map_err(|e| {
|
||||||
|
ErrorBuilder::new(EditorErrorCode::DocNameInvalid)
|
||||||
|
.msg(e)
|
||||||
|
.build()
|
||||||
|
})?
|
||||||
|
.0;
|
||||||
|
|
||||||
|
let view_id = DocViewId::parse(self.view_id)
|
||||||
|
.map_err(|e| {
|
||||||
|
ErrorBuilder::new(EditorErrorCode::DocViewIdInvalid)
|
||||||
|
.msg(e)
|
||||||
|
.build()
|
||||||
|
})?
|
||||||
|
.0;
|
||||||
|
|
||||||
|
Ok(CreateDocParams { view_id, name })
|
||||||
|
}
|
||||||
|
}
|
5
rust-lib/flowy-editor/src/entities/doc/mod.rs
Normal file
5
rust-lib/flowy-editor/src/entities/doc/mod.rs
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
mod doc_create;
|
||||||
|
mod doc_modify;
|
||||||
|
mod parser;
|
||||||
|
|
||||||
|
pub use doc_create::*;
|
12
rust-lib/flowy-editor/src/entities/doc/parser/doc_name.rs
Normal file
12
rust-lib/flowy-editor/src/entities/doc/parser/doc_name.rs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
#[derive(Debug)]
|
||||||
|
pub struct DocName(pub String);
|
||||||
|
|
||||||
|
impl DocName {
|
||||||
|
pub fn parse(s: String) -> Result<DocName, String> {
|
||||||
|
if s.trim().is_empty() {
|
||||||
|
return Err(format!("Doc name can not be empty or whitespace"));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self(s))
|
||||||
|
}
|
||||||
|
}
|
12
rust-lib/flowy-editor/src/entities/doc/parser/doc_view_id.rs
Normal file
12
rust-lib/flowy-editor/src/entities/doc/parser/doc_view_id.rs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
#[derive(Debug)]
|
||||||
|
pub struct DocViewId(pub String);
|
||||||
|
|
||||||
|
impl DocViewId {
|
||||||
|
pub fn parse(s: String) -> Result<DocViewId, String> {
|
||||||
|
if s.trim().is_empty() {
|
||||||
|
return Err(format!("Doc view id can not be empty or whitespace"));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self(s))
|
||||||
|
}
|
||||||
|
}
|
5
rust-lib/flowy-editor/src/entities/doc/parser/mod.rs
Normal file
5
rust-lib/flowy-editor/src/entities/doc/parser/mod.rs
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
mod doc_name;
|
||||||
|
mod doc_view_id;
|
||||||
|
|
||||||
|
pub use doc_name::*;
|
||||||
|
pub use doc_view_id::*;
|
1
rust-lib/flowy-editor/src/entities/mod.rs
Normal file
1
rust-lib/flowy-editor/src/entities/mod.rs
Normal file
@ -0,0 +1 @@
|
|||||||
|
pub mod doc;
|
85
rust-lib/flowy-editor/src/errors.rs
Normal file
85
rust-lib/flowy-editor/src/errors.rs
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
use derive_more::Display;
|
||||||
|
use flowy_derive::{ProtoBuf, ProtoBuf_Enum};
|
||||||
|
use flowy_dispatch::prelude::{EventResponse, ResponseBuilder};
|
||||||
|
use std::convert::TryInto;
|
||||||
|
|
||||||
|
#[derive(Debug, Default, Clone, ProtoBuf)]
|
||||||
|
pub struct EditorError {
|
||||||
|
#[pb(index = 1)]
|
||||||
|
pub code: UserErrorCode,
|
||||||
|
|
||||||
|
#[pb(index = 2)]
|
||||||
|
pub msg: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EditorError {
|
||||||
|
fn new(code: EditorErrorCode, msg: &str) -> Self {
|
||||||
|
Self {
|
||||||
|
code,
|
||||||
|
msg: msg.to_owned(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, ProtoBuf_Enum, Display, PartialEq, Eq)]
|
||||||
|
pub enum EditorErrorCode {
|
||||||
|
#[display(fmt = "Unknown")]
|
||||||
|
Unknown = 0,
|
||||||
|
|
||||||
|
#[display(fmt = "EditorDBInternalError")]
|
||||||
|
EditorDBInternalError = 1,
|
||||||
|
|
||||||
|
#[display(fmt = "DocNameInvalid")]
|
||||||
|
DocNameInvalid = 10,
|
||||||
|
|
||||||
|
#[display(fmt = "DocViewIdInvalid")]
|
||||||
|
DocViewIdInvalid = 11,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::default::Default for UserErrorCode {
|
||||||
|
fn default() -> Self { UserErrorCode::Unknown }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::convert::From<flowy_database::result::Error> for EditorError {
|
||||||
|
fn from(error: flowy_database::result::Error) -> Self {
|
||||||
|
ErrorBuilder::new(EditorErrorCode::EditorDBInternalError)
|
||||||
|
.error(error)
|
||||||
|
.build()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl flowy_dispatch::Error for EditorError {
|
||||||
|
fn as_response(&self) -> EventResponse {
|
||||||
|
let bytes: Vec<u8> = self.clone().try_into().unwrap();
|
||||||
|
ResponseBuilder::Err().data(bytes).build()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ErrorBuilder {
|
||||||
|
pub code: UserErrorCode,
|
||||||
|
pub msg: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ErrorBuilder {
|
||||||
|
pub fn new(code: EditorErrorCode) -> Self { ErrorBuilder { code, msg: None } }
|
||||||
|
|
||||||
|
pub fn msg<T>(mut self, msg: T) -> Self
|
||||||
|
where
|
||||||
|
T: Into<String>,
|
||||||
|
{
|
||||||
|
self.msg = Some(msg.into());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn error<T>(mut self, msg: T) -> Self
|
||||||
|
where
|
||||||
|
T: std::fmt::Debug,
|
||||||
|
{
|
||||||
|
self.msg = Some(format!("{:?}", msg));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build(mut self) -> EditorError {
|
||||||
|
EditorError::new(self.code, &self.msg.take().unwrap_or("".to_owned()))
|
||||||
|
}
|
||||||
|
}
|
0
rust-lib/flowy-editor/src/event.rs
Normal file
0
rust-lib/flowy-editor/src/event.rs
Normal file
117
rust-lib/flowy-editor/src/file_manager/file.rs
Normal file
117
rust-lib/flowy-editor/src/file_manager/file.rs
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
use std::{
|
||||||
|
ffi::OsString,
|
||||||
|
fs,
|
||||||
|
fs::File,
|
||||||
|
io,
|
||||||
|
io::{Read, Write},
|
||||||
|
path::{Path, PathBuf},
|
||||||
|
str,
|
||||||
|
time::SystemTime,
|
||||||
|
};
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
|
pub struct FileId(pub(crate) usize);
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub enum CharacterEncoding {
|
||||||
|
Utf8,
|
||||||
|
Utf8WithBom,
|
||||||
|
}
|
||||||
|
|
||||||
|
const UTF8_BOM: &str = "\u{feff}";
|
||||||
|
impl CharacterEncoding {
|
||||||
|
pub(crate) fn guess(s: &[u8]) -> Self {
|
||||||
|
if s.starts_with(UTF8_BOM.as_bytes()) {
|
||||||
|
CharacterEncoding::Utf8WithBom
|
||||||
|
} else {
|
||||||
|
CharacterEncoding::Utf8
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum FileError {
|
||||||
|
Io(io::Error, PathBuf),
|
||||||
|
UnknownEncoding(PathBuf),
|
||||||
|
HasChanged(PathBuf),
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct FileInfo {
|
||||||
|
pub path: PathBuf,
|
||||||
|
pub modify_time: Option<SystemTime>,
|
||||||
|
pub has_changed: bool,
|
||||||
|
pub encoding: CharacterEncoding,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn try_load_file<P>(path: P) -> Result<(String, FileInfo), FileError>
|
||||||
|
where
|
||||||
|
P: AsRef<Path>,
|
||||||
|
{
|
||||||
|
let mut f =
|
||||||
|
File::open(path.as_ref()).map_err(|e| FileError::Io(e, path.as_ref().to_owned()))?;
|
||||||
|
let mut bytes = Vec::new();
|
||||||
|
f.read_to_end(&mut bytes)
|
||||||
|
.map_err(|e| FileError::Io(e, path.as_ref().to_owned()))?;
|
||||||
|
|
||||||
|
let encoding = CharacterEncoding::guess(&bytes);
|
||||||
|
let s = try_decode(bytes, encoding, path.as_ref())?;
|
||||||
|
let info = FileInfo {
|
||||||
|
encoding,
|
||||||
|
path: path.as_ref().to_owned(),
|
||||||
|
modify_time: get_mod_time(&path),
|
||||||
|
has_changed: false,
|
||||||
|
};
|
||||||
|
Ok((s, info))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_mod_time<P: AsRef<Path>>(path: P) -> Option<SystemTime> {
|
||||||
|
File::open(path)
|
||||||
|
.and_then(|f| f.metadata())
|
||||||
|
.and_then(|meta| meta.modified())
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn try_save(
|
||||||
|
path: &Path,
|
||||||
|
text: &str,
|
||||||
|
encoding: CharacterEncoding,
|
||||||
|
_file_info: Option<&FileInfo>,
|
||||||
|
) -> io::Result<()> {
|
||||||
|
let tmp_extension = path.extension().map_or_else(
|
||||||
|
|| OsString::from("swp"),
|
||||||
|
|ext| {
|
||||||
|
let mut ext = ext.to_os_string();
|
||||||
|
ext.push(".swp");
|
||||||
|
ext
|
||||||
|
},
|
||||||
|
);
|
||||||
|
let tmp_path = &path.with_extension(tmp_extension);
|
||||||
|
|
||||||
|
let mut f = File::create(tmp_path)?;
|
||||||
|
match encoding {
|
||||||
|
CharacterEncoding::Utf8WithBom => f.write_all(UTF8_BOM.as_bytes())?,
|
||||||
|
CharacterEncoding::Utf8 => (),
|
||||||
|
}
|
||||||
|
|
||||||
|
f.write_all(text.as_bytes())?;
|
||||||
|
fs::rename(tmp_path, path)?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn try_decode(
|
||||||
|
bytes: Vec<u8>,
|
||||||
|
encoding: CharacterEncoding,
|
||||||
|
path: &Path,
|
||||||
|
) -> Result<String, FileError> {
|
||||||
|
match encoding {
|
||||||
|
CharacterEncoding::Utf8 => {
|
||||||
|
Ok(String::from(str::from_utf8(&bytes).map_err(|_e| {
|
||||||
|
FileError::UnknownEncoding(path.to_owned())
|
||||||
|
})?))
|
||||||
|
},
|
||||||
|
CharacterEncoding::Utf8WithBom => {
|
||||||
|
let s = String::from_utf8(bytes)
|
||||||
|
.map_err(|_e| FileError::UnknownEncoding(path.to_owned()))?;
|
||||||
|
Ok(String::from(&s[UTF8_BOM.len()..]))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
39
rust-lib/flowy-editor/src/file_manager/manager.rs
Normal file
39
rust-lib/flowy-editor/src/file_manager/manager.rs
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
use crate::file_manager::file::*;
|
||||||
|
use std::{
|
||||||
|
collections::HashMap,
|
||||||
|
path::{Path, PathBuf},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct FileManager {
|
||||||
|
open_files: HashMap<PathBuf, FileId>,
|
||||||
|
file_info: HashMap<FileId, FileInfo>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FileManager {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
open_files: HashMap::new(),
|
||||||
|
file_info: HashMap::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_info(&self, id: FileId) -> Option<&FileInfo> { self.file_info.get(&id) }
|
||||||
|
|
||||||
|
pub fn get_editor(&self, path: &Path) -> Option<FileId> { self.open_files.get(path).cloned() }
|
||||||
|
|
||||||
|
pub fn open(&mut self, path: &Path, id: FileId) -> Result<String, FileError> {
|
||||||
|
if !path.exists() {
|
||||||
|
return Ok("".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
let (s, info) = try_load_file(path)?;
|
||||||
|
self.open_files.insert(path.to_owned(), id);
|
||||||
|
Ok(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn close(&mut self, id: FileId) {
|
||||||
|
if let Some(info) = self.file_info.remove(&id) {
|
||||||
|
self.open_files.remove(&info.path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
2
rust-lib/flowy-editor/src/file_manager/mod.rs
Normal file
2
rust-lib/flowy-editor/src/file_manager/mod.rs
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
mod file;
|
||||||
|
mod manager;
|
0
rust-lib/flowy-editor/src/handlers/mod.rs
Normal file
0
rust-lib/flowy-editor/src/handlers/mod.rs
Normal file
6
rust-lib/flowy-editor/src/lib.rs
Normal file
6
rust-lib/flowy-editor/src/lib.rs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
mod entities;
|
||||||
|
mod errors;
|
||||||
|
mod event;
|
||||||
|
mod file_manager;
|
||||||
|
mod handlers;
|
||||||
|
mod module;
|
0
rust-lib/flowy-editor/src/module.rs
Normal file
0
rust-lib/flowy-editor/src/module.rs
Normal file
Loading…
x
Reference in New Issue
Block a user