|
@@ -1,7 +1,9 @@
|
|
|
use crate::entities::{CellIdentifier, CellIdentifierPayload};
|
|
|
use crate::impl_type_option;
|
|
|
use crate::services::field::{BoxTypeOptionBuilder, TypeOptionBuilder};
|
|
|
-use crate::services::row::{CellContentChangeset, CellDataOperation, DecodedCellData, TypeOptionCellData};
|
|
|
+use crate::services::row::{
|
|
|
+ CellContentChangeset, CellDataOperation, DecodedCellData, EncodedCellData, TypeOptionCellData,
|
|
|
+};
|
|
|
use bytes::Bytes;
|
|
|
use chrono::format::strftime::StrftimeItems;
|
|
|
use chrono::NaiveDateTime;
|
|
@@ -90,10 +92,10 @@ impl DateTypeOption {
|
|
|
|
|
|
let serde_cell_data = DateCellDataSerde::from_str(&result.unwrap().data)?;
|
|
|
let date = self.decode_cell_data_from_timestamp(&serde_cell_data).content;
|
|
|
- let time = serde_cell_data.time.unwrap_or("".to_owned());
|
|
|
+ let time = serde_cell_data.time.unwrap_or_else(|| "".to_owned());
|
|
|
let timestamp = serde_cell_data.timestamp;
|
|
|
|
|
|
- return Ok(DateCellData { date, time, timestamp });
|
|
|
+ Ok(DateCellData { date, time, timestamp })
|
|
|
}
|
|
|
|
|
|
fn decode_cell_data_from_timestamp(&self, serde_cell_data: &DateCellDataSerde) -> DecodedCellData {
|
|
@@ -102,7 +104,7 @@ impl DateTypeOption {
|
|
|
}
|
|
|
|
|
|
let cell_content = self.today_desc_from_timestamp(serde_cell_data.timestamp, &serde_cell_data.time);
|
|
|
- return DecodedCellData::new(serde_cell_data.timestamp.to_string(), cell_content);
|
|
|
+ DecodedCellData::new(serde_cell_data.timestamp.to_string(), cell_content)
|
|
|
}
|
|
|
|
|
|
fn timestamp_from_utc_with_time(
|
|
@@ -131,29 +133,30 @@ impl DateTypeOption {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- return Ok(utc.timestamp());
|
|
|
+ Ok(utc.timestamp())
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-impl CellDataOperation<DateCellDataSerde> for DateTypeOption {
|
|
|
- fn decode_cell_data<T: Into<TypeOptionCellData>>(
|
|
|
+impl CellDataOperation<EncodedCellData<DateCellDataSerde>, DateCellDataSerde> for DateTypeOption {
|
|
|
+ fn decode_cell_data<T>(
|
|
|
&self,
|
|
|
- type_option_cell_data: T,
|
|
|
+ encoded_data: T,
|
|
|
decoded_field_type: &FieldType,
|
|
|
_field_meta: &FieldMeta,
|
|
|
- ) -> DecodedCellData {
|
|
|
- let type_option_cell_data = type_option_cell_data.into();
|
|
|
+ ) -> FlowyResult<DecodedCellData>
|
|
|
+ where
|
|
|
+ T: Into<EncodedCellData<DateCellDataSerde>>,
|
|
|
+ {
|
|
|
// Return default data if the type_option_cell_data is not FieldType::DateTime.
|
|
|
// It happens when switching from one field to another.
|
|
|
// For example:
|
|
|
// FieldType::RichText -> FieldType::DateTime, it will display empty content on the screen.
|
|
|
if !decoded_field_type.is_date() {
|
|
|
- return DecodedCellData::default();
|
|
|
- }
|
|
|
- match DateCellDataSerde::from_str(&type_option_cell_data.data) {
|
|
|
- Ok(serde_cell_data) => self.decode_cell_data_from_timestamp(&serde_cell_data),
|
|
|
- Err(_) => DecodedCellData::default(),
|
|
|
+ return Ok(DecodedCellData::default());
|
|
|
}
|
|
|
+
|
|
|
+ let encoded_data = encoded_data.into().try_into_inner()?;
|
|
|
+ Ok(self.decode_cell_data_from_timestamp(&encoded_data))
|
|
|
}
|
|
|
|
|
|
fn apply_changeset<C>(&self, changeset: C, _cell_meta: Option<CellMeta>) -> Result<DateCellDataSerde, FlowyError>
|
|
@@ -308,22 +311,26 @@ impl DateCellDataSerde {
|
|
|
fn new(timestamp: i64, time: Option<String>, time_format: &TimeFormat) -> Self {
|
|
|
Self {
|
|
|
timestamp,
|
|
|
- time: Some(time.unwrap_or(default_time_str(time_format))),
|
|
|
+ time: Some(time.unwrap_or_else(|| default_time_str(time_format))),
|
|
|
}
|
|
|
}
|
|
|
|
|
|
pub(crate) fn from_timestamp(timestamp: i64, time: Option<String>) -> Self {
|
|
|
Self { timestamp, time }
|
|
|
}
|
|
|
+}
|
|
|
+
|
|
|
+impl FromStr for DateCellDataSerde {
|
|
|
+ type Err = FlowyError;
|
|
|
|
|
|
- fn from_str(s: &str) -> FlowyResult<Self> {
|
|
|
+ fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
serde_json::from_str::<DateCellDataSerde>(s).map_err(internal_error)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
impl ToString for DateCellDataSerde {
|
|
|
fn to_string(&self) -> String {
|
|
|
- serde_json::to_string(&self).unwrap_or("".to_string())
|
|
|
+ serde_json::to_string(&self).unwrap_or_else(|_| "".to_string())
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -410,11 +417,11 @@ impl std::convert::From<DateCellContentChangeset> for CellContentChangeset {
|
|
|
#[cfg(test)]
|
|
|
mod tests {
|
|
|
use crate::services::field::FieldBuilder;
|
|
|
- use crate::services::field::{
|
|
|
- DateCellContentChangeset, DateCellData, DateCellDataSerde, DateFormat, DateTypeOption, TimeFormat,
|
|
|
+ use crate::services::field::{DateCellContentChangeset, DateCellDataSerde, DateFormat, DateTypeOption, TimeFormat};
|
|
|
+ use crate::services::row::{
|
|
|
+ apply_cell_data_changeset, decode_cell_data_from_type_option_cell_data, CellDataOperation, EncodedCellData,
|
|
|
};
|
|
|
- use crate::services::row::{apply_cell_data_changeset, decode_cell_data, CellDataOperation, TypeOptionCellData};
|
|
|
- use flowy_grid_data_model::entities::FieldType;
|
|
|
+ use flowy_grid_data_model::entities::{FieldMeta, FieldType};
|
|
|
use strum::IntoEnumIterator;
|
|
|
|
|
|
#[test]
|
|
@@ -422,9 +429,7 @@ mod tests {
|
|
|
let field_meta = FieldBuilder::from_field_type(&FieldType::Number).build();
|
|
|
let data = apply_cell_data_changeset("1e", None, &field_meta).unwrap();
|
|
|
assert_eq!(
|
|
|
- decode_cell_data(data, &field_meta, &field_meta.field_type)
|
|
|
- .unwrap()
|
|
|
- .content,
|
|
|
+ decode_cell_data_from_type_option_cell_data(data, &field_meta, &field_meta.field_type).content,
|
|
|
"".to_owned()
|
|
|
);
|
|
|
}
|
|
@@ -432,32 +437,44 @@ mod tests {
|
|
|
#[test]
|
|
|
fn date_description_date_format_test() {
|
|
|
let mut type_option = DateTypeOption::default();
|
|
|
- let field_meta = FieldBuilder::from_field_type(&FieldType::Number).build();
|
|
|
+ let field_meta = FieldBuilder::from_field_type(&FieldType::DateTime).build();
|
|
|
for date_format in DateFormat::iter() {
|
|
|
type_option.date_format = date_format;
|
|
|
match date_format {
|
|
|
DateFormat::Friendly => {
|
|
|
assert_eq!(
|
|
|
"Mar 14,2022".to_owned(),
|
|
|
- type_option.decode_cell_data(data(1647251762), &field_meta).content
|
|
|
+ type_option
|
|
|
+ .decode_cell_data(data(1647251762), &FieldType::DateTime, &field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content
|
|
|
);
|
|
|
}
|
|
|
DateFormat::US => {
|
|
|
assert_eq!(
|
|
|
"2022/03/14".to_owned(),
|
|
|
- type_option.decode_cell_data(data(1647251762), &field_meta).content
|
|
|
+ type_option
|
|
|
+ .decode_cell_data(data(1647251762), &FieldType::DateTime, &field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content
|
|
|
);
|
|
|
}
|
|
|
DateFormat::ISO => {
|
|
|
assert_eq!(
|
|
|
"2022-03-14".to_owned(),
|
|
|
- type_option.decode_cell_data(data(1647251762), &field_meta).content
|
|
|
+ type_option
|
|
|
+ .decode_cell_data(data(1647251762), &FieldType::DateTime, &field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content
|
|
|
);
|
|
|
}
|
|
|
DateFormat::Local => {
|
|
|
assert_eq!(
|
|
|
"2022/03/14".to_owned(),
|
|
|
- type_option.decode_cell_data(data(1647251762), &field_meta).content
|
|
|
+ type_option
|
|
|
+ .decode_cell_data(data(1647251762), &FieldType::DateTime, &field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content
|
|
|
);
|
|
|
}
|
|
|
}
|
|
@@ -467,7 +484,7 @@ mod tests {
|
|
|
#[test]
|
|
|
fn date_description_time_format_test() {
|
|
|
let mut type_option = DateTypeOption::default();
|
|
|
- let field_meta = FieldBuilder::from_field_type(&FieldType::Number).build();
|
|
|
+ let field_meta = FieldBuilder::from_field_type(&FieldType::DateTime).build();
|
|
|
for time_format in TimeFormat::iter() {
|
|
|
type_option.time_format = time_format;
|
|
|
match time_format {
|
|
@@ -478,7 +495,10 @@ mod tests {
|
|
|
);
|
|
|
assert_eq!(
|
|
|
"Mar 14,2022".to_owned(),
|
|
|
- type_option.decode_cell_data(data(1647251762), &field_meta).content
|
|
|
+ type_option
|
|
|
+ .decode_cell_data(data(1647251762), &FieldType::DateTime, &field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content
|
|
|
);
|
|
|
}
|
|
|
TimeFormat::TwelveHour => {
|
|
@@ -488,7 +508,10 @@ mod tests {
|
|
|
);
|
|
|
assert_eq!(
|
|
|
"Mar 14,2022".to_owned(),
|
|
|
- type_option.decode_cell_data(data(1647251762), &field_meta).content
|
|
|
+ type_option
|
|
|
+ .decode_cell_data(data(1647251762), &FieldType::DateTime, &field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content
|
|
|
);
|
|
|
}
|
|
|
}
|
|
@@ -498,124 +521,171 @@ mod tests {
|
|
|
#[test]
|
|
|
fn date_description_time_format_test2() {
|
|
|
let mut type_option = DateTypeOption::default();
|
|
|
- let field_meta = FieldBuilder::from_field_type(&FieldType::Number).build();
|
|
|
+ let field_type = FieldType::DateTime;
|
|
|
+ let field_meta = FieldBuilder::from_field_type(&field_type).build();
|
|
|
+
|
|
|
for time_format in TimeFormat::iter() {
|
|
|
type_option.time_format = time_format;
|
|
|
type_option.include_time = true;
|
|
|
match time_format {
|
|
|
TimeFormat::TwentyFourHour => {
|
|
|
- let changeset = DateCellContentChangeset {
|
|
|
- date: Some(1653609600.to_string()),
|
|
|
- time: None,
|
|
|
- };
|
|
|
- let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- assert_eq!("May 27,2022 00:00".to_owned(), content);
|
|
|
-
|
|
|
- let changeset = DateCellContentChangeset {
|
|
|
- date: Some(1653609600.to_string()),
|
|
|
- time: Some("23:00".to_owned()),
|
|
|
- };
|
|
|
-
|
|
|
- let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- assert_eq!("May 27,2022 23:00".to_owned(), content);
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(1653609600.to_string()),
|
|
|
+ time: None,
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 00:00",
|
|
|
+ );
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(1653609600.to_string()),
|
|
|
+ time: Some("23:00".to_owned()),
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 23:00",
|
|
|
+ );
|
|
|
}
|
|
|
TimeFormat::TwelveHour => {
|
|
|
- let changeset = DateCellContentChangeset {
|
|
|
- date: Some(1653609600.to_string()),
|
|
|
- time: None,
|
|
|
- };
|
|
|
- let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- assert_eq!("May 27,2022 12:00 AM".to_owned(), content);
|
|
|
-
|
|
|
- let changeset = DateCellContentChangeset {
|
|
|
- date: Some(1653609600.to_string()),
|
|
|
- time: Some("".to_owned()),
|
|
|
- };
|
|
|
- let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- assert_eq!("May 27,2022".to_owned(), content);
|
|
|
-
|
|
|
- let changeset = DateCellContentChangeset {
|
|
|
- date: Some(1653609600.to_string()),
|
|
|
- time: Some("11:23 pm".to_owned()),
|
|
|
- };
|
|
|
- let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- assert_eq!("May 27,2022 11:23 PM".to_owned(), content);
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(1653609600.to_string()),
|
|
|
+ time: None,
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 12:00 AM",
|
|
|
+ );
|
|
|
+
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(1653609600.to_string()),
|
|
|
+ time: Some("".to_owned()),
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022",
|
|
|
+ );
|
|
|
+
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(1653609600.to_string()),
|
|
|
+ time: Some("11:23 pm".to_owned()),
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 11:23 PM",
|
|
|
+ );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- // #[test]
|
|
|
- // fn date_description_apply_changeset_test() {
|
|
|
- // let mut type_option = DateTypeOption::default();
|
|
|
- // let field_meta = FieldBuilder::from_field_type(&FieldType::Number).build();
|
|
|
- // let date_timestamp = "1653609600".to_owned();
|
|
|
- //
|
|
|
- // let changeset = DateCellContentChangeset {
|
|
|
- // date: Some(date_timestamp.clone()),
|
|
|
- // time: None,
|
|
|
- // };
|
|
|
- // let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- // let content = type_option.decode_cell_data(result.clone(), &field_meta).content;
|
|
|
- // assert_eq!(content, "May 27,2022".to_owned());
|
|
|
- //
|
|
|
- // type_option.include_time = true;
|
|
|
- // let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- // assert_eq!(content, "May 27,2022 00:00".to_owned());
|
|
|
- //
|
|
|
- // let changeset = DateCellContentChangeset {
|
|
|
- // date: Some(date_timestamp.clone()),
|
|
|
- // time: Some("1:00".to_owned()),
|
|
|
- // };
|
|
|
- // let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- // let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- // assert_eq!(content, "May 27,2022 01:00".to_owned());
|
|
|
- //
|
|
|
- // let changeset = DateCellContentChangeset {
|
|
|
- // date: Some(date_timestamp),
|
|
|
- // time: Some("1:00 am".to_owned()),
|
|
|
- // };
|
|
|
- // type_option.time_format = TimeFormat::TwelveHour;
|
|
|
- // let result = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- // let content = type_option.decode_cell_data(result, &field_meta).content;
|
|
|
- // assert_eq!(content, "May 27,2022 01:00 AM".to_owned());
|
|
|
- // }
|
|
|
- //
|
|
|
- // #[test]
|
|
|
- // #[should_panic]
|
|
|
- // fn date_description_apply_changeset_error_test() {
|
|
|
- // let mut type_option = DateTypeOption::default();
|
|
|
- // type_option.include_time = true;
|
|
|
- // let field_meta = FieldBuilder::from_field_type(&FieldType::Number).build();
|
|
|
- // let date_timestamp = "1653609600".to_owned();
|
|
|
- //
|
|
|
- // let changeset = DateCellContentChangeset {
|
|
|
- // date: Some(date_timestamp.clone()),
|
|
|
- // time: Some("1:a0".to_owned()),
|
|
|
- // };
|
|
|
- // let _ = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- //
|
|
|
- // let changeset = DateCellContentChangeset {
|
|
|
- // date: Some(date_timestamp.clone()),
|
|
|
- // time: Some("1:".to_owned()),
|
|
|
- // };
|
|
|
- // let _ = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
- // }
|
|
|
- //
|
|
|
- // #[test]
|
|
|
- // #[should_panic]
|
|
|
- // fn date_description_invalid_data_test() {
|
|
|
- // let type_option = DateTypeOption::default();
|
|
|
- // type_option.apply_changeset("he", None).unwrap();
|
|
|
- // }
|
|
|
- //
|
|
|
- fn data(s: i64) -> TypeOptionCellData {
|
|
|
- let json = serde_json::to_string(&DateCellDataSerde::from_timestamp(s, None)).unwrap();
|
|
|
- TypeOptionCellData::new(&json, FieldType::DateTime)
|
|
|
+ #[test]
|
|
|
+ fn date_description_apply_changeset_test() {
|
|
|
+ let mut type_option = DateTypeOption::default();
|
|
|
+ let field_type = FieldType::DateTime;
|
|
|
+ let field_meta = FieldBuilder::from_field_type(&field_type).build();
|
|
|
+ let date_timestamp = "1653609600".to_owned();
|
|
|
+
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(date_timestamp.clone()),
|
|
|
+ time: None,
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022",
|
|
|
+ );
|
|
|
+
|
|
|
+ type_option.include_time = true;
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(date_timestamp.clone()),
|
|
|
+ time: None,
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 00:00",
|
|
|
+ );
|
|
|
+
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(date_timestamp.clone()),
|
|
|
+ time: Some("1:00".to_owned()),
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 01:00",
|
|
|
+ );
|
|
|
+
|
|
|
+ type_option.time_format = TimeFormat::TwelveHour;
|
|
|
+ assert_result(
|
|
|
+ &type_option,
|
|
|
+ DateCellContentChangeset {
|
|
|
+ date: Some(date_timestamp),
|
|
|
+ time: Some("1:00 am".to_owned()),
|
|
|
+ },
|
|
|
+ &field_type,
|
|
|
+ &field_meta,
|
|
|
+ "May 27,2022 01:00 AM",
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ #[test]
|
|
|
+ #[should_panic]
|
|
|
+ fn date_description_apply_changeset_error_test() {
|
|
|
+ let mut type_option = DateTypeOption::default();
|
|
|
+ type_option.include_time = true;
|
|
|
+ let _field_meta = FieldBuilder::from_field_type(&FieldType::DateTime).build();
|
|
|
+ let date_timestamp = "1653609600".to_owned();
|
|
|
+
|
|
|
+ let changeset = DateCellContentChangeset {
|
|
|
+ date: Some(date_timestamp.clone()),
|
|
|
+ time: Some("1:a0".to_owned()),
|
|
|
+ };
|
|
|
+ let _ = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
+
|
|
|
+ let changeset = DateCellContentChangeset {
|
|
|
+ date: Some(date_timestamp),
|
|
|
+ time: Some("1:".to_owned()),
|
|
|
+ };
|
|
|
+ let _ = type_option.apply_changeset(changeset, None).unwrap();
|
|
|
+ }
|
|
|
+
|
|
|
+ #[test]
|
|
|
+ #[should_panic]
|
|
|
+ fn date_description_invalid_data_test() {
|
|
|
+ let type_option = DateTypeOption::default();
|
|
|
+ type_option.apply_changeset("he", None).unwrap();
|
|
|
+ }
|
|
|
+
|
|
|
+ fn data(s: i64) -> String {
|
|
|
+ serde_json::to_string(&DateCellDataSerde::from_timestamp(s, None)).unwrap()
|
|
|
+ }
|
|
|
+
|
|
|
+ fn assert_result(
|
|
|
+ type_option: &DateTypeOption,
|
|
|
+ changeset: DateCellContentChangeset,
|
|
|
+ field_type: &FieldType,
|
|
|
+ field_meta: &FieldMeta,
|
|
|
+ expected: &str,
|
|
|
+ ) {
|
|
|
+ let encoded_data = EncodedCellData(Some(type_option.apply_changeset(changeset, None).unwrap()));
|
|
|
+ let content = type_option
|
|
|
+ .decode_cell_data(encoded_data, field_type, field_meta)
|
|
|
+ .unwrap()
|
|
|
+ .content;
|
|
|
+ assert_eq!(expected.to_owned(), content);
|
|
|
}
|
|
|
}
|