summaryrefslogtreecommitdiff
path: root/Userland/Applications/Spreadsheet
diff options
context:
space:
mode:
authorAli Mohammad Pur <ali.mpfard@gmail.com>2021-11-21 05:08:00 +0330
committerAli Mohammad Pur <Ali.mpfard@gmail.com>2021-12-12 14:49:49 +0330
commit5f1a34bba3a0eb81b8d4474daffe8e837828277e (patch)
tree503710d8cee210aca8394875321957f2a8312299 /Userland/Applications/Spreadsheet
parent235eb0b1adba5920ce914e5a908b5940c2ddf9f8 (diff)
downloadserenity-5f1a34bba3a0eb81b8d4474daffe8e837828277e.zip
Spreadsheet: Avoid using Value.to_string_without_side_effects()
We should use .to_string() and handle the possible exceptions. This makes the displayed cell contents so much more informative than '[object Object]' :^)
Diffstat (limited to 'Userland/Applications/Spreadsheet')
-rw-r--r--Userland/Applications/Spreadsheet/Cell.cpp4
-rw-r--r--Userland/Applications/Spreadsheet/Cell.h4
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Date.cpp10
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Date.h4
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Identity.cpp6
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Identity.h4
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Numeric.cpp12
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Numeric.h4
-rw-r--r--Userland/Applications/Spreadsheet/CellType/String.cpp8
-rw-r--r--Userland/Applications/Spreadsheet/CellType/String.h4
-rw-r--r--Userland/Applications/Spreadsheet/CellType/Type.h4
-rw-r--r--Userland/Applications/Spreadsheet/Spreadsheet.cpp7
-rw-r--r--Userland/Applications/Spreadsheet/SpreadsheetModel.cpp56
13 files changed, 73 insertions, 54 deletions
diff --git a/Userland/Applications/Spreadsheet/Cell.cpp b/Userland/Applications/Spreadsheet/Cell.cpp
index e3bd310b42..176887eb3b 100644
--- a/Userland/Applications/Spreadsheet/Cell.cpp
+++ b/Userland/Applications/Spreadsheet/Cell.cpp
@@ -74,12 +74,12 @@ const CellType& Cell::type() const
return *CellType::get_by_name("Identity");
}
-String Cell::typed_display() const
+JS::ThrowCompletionOr<String> Cell::typed_display() const
{
return type().display(const_cast<Cell&>(*this), m_type_metadata);
}
-JS::Value Cell::typed_js_data() const
+JS::ThrowCompletionOr<JS::Value> Cell::typed_js_data() const
{
return type().js_value(const_cast<Cell&>(*this), m_type_metadata);
}
diff --git a/Userland/Applications/Spreadsheet/Cell.h b/Userland/Applications/Spreadsheet/Cell.h
index d2d3541c1a..1e8830084e 100644
--- a/Userland/Applications/Spreadsheet/Cell.h
+++ b/Userland/Applications/Spreadsheet/Cell.h
@@ -79,8 +79,8 @@ struct Cell : public Weakable<Cell> {
m_conditional_formats = move(fmts);
}
- String typed_display() const;
- JS::Value typed_js_data() const;
+ JS::ThrowCompletionOr<String> typed_display() const;
+ JS::ThrowCompletionOr<JS::Value> typed_js_data() const;
const CellType& type() const;
const CellTypeMetadata& type_metadata() const { return m_type_metadata; }
diff --git a/Userland/Applications/Spreadsheet/CellType/Date.cpp b/Userland/Applications/Spreadsheet/CellType/Date.cpp
index 9c257dab65..9714ebb47d 100644
--- a/Userland/Applications/Spreadsheet/CellType/Date.cpp
+++ b/Userland/Applications/Spreadsheet/CellType/Date.cpp
@@ -21,7 +21,7 @@ DateCell::~DateCell()
{
}
-String DateCell::display(Cell& cell, const CellTypeMetadata& metadata) const
+JS::ThrowCompletionOr<String> DateCell::display(Cell& cell, const CellTypeMetadata& metadata) const
{
ScopeGuard propagate_exception { [&cell] {
if (auto exc = cell.sheet().interpreter().exception()) {
@@ -29,8 +29,8 @@ String DateCell::display(Cell& cell, const CellTypeMetadata& metadata) const
cell.set_exception(exc);
}
} };
- auto timestamp = js_value(cell, metadata);
- auto string = Core::DateTime::from_timestamp(TRY_OR_DISCARD(timestamp.to_i32(cell.sheet().global_object()))).to_string(metadata.format.is_empty() ? "%Y-%m-%d %H:%M:%S" : metadata.format.characters());
+ auto timestamp = TRY(js_value(cell, metadata));
+ auto string = Core::DateTime::from_timestamp(TRY(timestamp.to_i32(cell.sheet().global_object()))).to_string(metadata.format.is_empty() ? "%Y-%m-%d %H:%M:%S" : metadata.format.characters());
if (metadata.length >= 0)
return string.substring(0, metadata.length);
@@ -38,10 +38,10 @@ String DateCell::display(Cell& cell, const CellTypeMetadata& metadata) const
return string;
}
-JS::Value DateCell::js_value(Cell& cell, const CellTypeMetadata&) const
+JS::ThrowCompletionOr<JS::Value> DateCell::js_value(Cell& cell, const CellTypeMetadata&) const
{
auto js_data = cell.js_data();
- auto value = TRY_OR_DISCARD(js_data.to_double(cell.sheet().global_object()));
+ auto value = TRY(js_data.to_double(cell.sheet().global_object()));
return JS::Value(value / 1000); // Turn it to seconds
}
diff --git a/Userland/Applications/Spreadsheet/CellType/Date.h b/Userland/Applications/Spreadsheet/CellType/Date.h
index 6e613a4380..b39583c371 100644
--- a/Userland/Applications/Spreadsheet/CellType/Date.h
+++ b/Userland/Applications/Spreadsheet/CellType/Date.h
@@ -15,8 +15,8 @@ class DateCell : public CellType {
public:
DateCell();
virtual ~DateCell() override;
- virtual String display(Cell&, const CellTypeMetadata&) const override;
- virtual JS::Value js_value(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<String> display(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<JS::Value> js_value(Cell&, const CellTypeMetadata&) const override;
};
}
diff --git a/Userland/Applications/Spreadsheet/CellType/Identity.cpp b/Userland/Applications/Spreadsheet/CellType/Identity.cpp
index c93ea23c4e..70a43c8be0 100644
--- a/Userland/Applications/Spreadsheet/CellType/Identity.cpp
+++ b/Userland/Applications/Spreadsheet/CellType/Identity.cpp
@@ -19,12 +19,12 @@ IdentityCell::~IdentityCell()
{
}
-String IdentityCell::display(Cell& cell, const CellTypeMetadata&) const
+JS::ThrowCompletionOr<String> IdentityCell::display(Cell& cell, const CellTypeMetadata&) const
{
- return cell.js_data().to_string_without_side_effects();
+ return cell.js_data().to_string(cell.sheet().global_object());
}
-JS::Value IdentityCell::js_value(Cell& cell, const CellTypeMetadata&) const
+JS::ThrowCompletionOr<JS::Value> IdentityCell::js_value(Cell& cell, const CellTypeMetadata&) const
{
return cell.js_data();
}
diff --git a/Userland/Applications/Spreadsheet/CellType/Identity.h b/Userland/Applications/Spreadsheet/CellType/Identity.h
index 659ee270a1..5811a7305e 100644
--- a/Userland/Applications/Spreadsheet/CellType/Identity.h
+++ b/Userland/Applications/Spreadsheet/CellType/Identity.h
@@ -15,8 +15,8 @@ class IdentityCell : public CellType {
public:
IdentityCell();
virtual ~IdentityCell() override;
- virtual String display(Cell&, const CellTypeMetadata&) const override;
- virtual JS::Value js_value(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<String> display(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<JS::Value> js_value(Cell&, const CellTypeMetadata&) const override;
};
}
diff --git a/Userland/Applications/Spreadsheet/CellType/Numeric.cpp b/Userland/Applications/Spreadsheet/CellType/Numeric.cpp
index 7b8f098a2a..9ebe1d8aa9 100644
--- a/Userland/Applications/Spreadsheet/CellType/Numeric.cpp
+++ b/Userland/Applications/Spreadsheet/CellType/Numeric.cpp
@@ -21,7 +21,7 @@ NumericCell::~NumericCell()
{
}
-String NumericCell::display(Cell& cell, const CellTypeMetadata& metadata) const
+JS::ThrowCompletionOr<String> NumericCell::display(Cell& cell, const CellTypeMetadata& metadata) const
{
ScopeGuard propagate_exception { [&cell] {
if (auto exc = cell.sheet().interpreter().exception()) {
@@ -29,12 +29,12 @@ String NumericCell::display(Cell& cell, const CellTypeMetadata& metadata) const
cell.set_exception(exc);
}
} };
- auto value = js_value(cell, metadata);
+ auto value = TRY(js_value(cell, metadata));
String string;
if (metadata.format.is_empty())
- string = value.to_string_without_side_effects();
+ string = TRY(value.to_string(cell.sheet().global_object()));
else
- string = format_double(metadata.format.characters(), TRY_OR_DISCARD(value.to_double(cell.sheet().global_object())));
+ string = format_double(metadata.format.characters(), TRY(value.to_double(cell.sheet().global_object())));
if (metadata.length >= 0)
return string.substring(0, metadata.length);
@@ -42,7 +42,7 @@ String NumericCell::display(Cell& cell, const CellTypeMetadata& metadata) const
return string;
}
-JS::Value NumericCell::js_value(Cell& cell, const CellTypeMetadata&) const
+JS::ThrowCompletionOr<JS::Value> NumericCell::js_value(Cell& cell, const CellTypeMetadata&) const
{
ScopeGuard propagate_exception { [&cell] {
if (auto exc = cell.sheet().interpreter().exception()) {
@@ -50,7 +50,7 @@ JS::Value NumericCell::js_value(Cell& cell, const CellTypeMetadata&) const
cell.set_exception(exc);
}
} };
- return TRY_OR_DISCARD(cell.js_data().to_number(cell.sheet().global_object()));
+ return cell.js_data().to_number(cell.sheet().global_object());
}
}
diff --git a/Userland/Applications/Spreadsheet/CellType/Numeric.h b/Userland/Applications/Spreadsheet/CellType/Numeric.h
index eb5a991adc..ae49204e3d 100644
--- a/Userland/Applications/Spreadsheet/CellType/Numeric.h
+++ b/Userland/Applications/Spreadsheet/CellType/Numeric.h
@@ -15,8 +15,8 @@ class NumericCell : public CellType {
public:
NumericCell();
virtual ~NumericCell() override;
- virtual String display(Cell&, const CellTypeMetadata&) const override;
- virtual JS::Value js_value(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<String> display(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<JS::Value> js_value(Cell&, const CellTypeMetadata&) const override;
};
}
diff --git a/Userland/Applications/Spreadsheet/CellType/String.cpp b/Userland/Applications/Spreadsheet/CellType/String.cpp
index fb1f360fc1..76a9be8782 100644
--- a/Userland/Applications/Spreadsheet/CellType/String.cpp
+++ b/Userland/Applications/Spreadsheet/CellType/String.cpp
@@ -19,18 +19,18 @@ StringCell::~StringCell()
{
}
-String StringCell::display(Cell& cell, const CellTypeMetadata& metadata) const
+JS::ThrowCompletionOr<String> StringCell::display(Cell& cell, const CellTypeMetadata& metadata) const
{
- auto string = cell.js_data().to_string_without_side_effects();
+ auto string = TRY(cell.js_data().to_string(cell.sheet().global_object()));
if (metadata.length >= 0)
return string.substring(0, metadata.length);
return string;
}
-JS::Value StringCell::js_value(Cell& cell, const CellTypeMetadata& metadata) const
+JS::ThrowCompletionOr<JS::Value> StringCell::js_value(Cell& cell, const CellTypeMetadata& metadata) const
{
- auto string = display(cell, metadata);
+ auto string = TRY(display(cell, metadata));
return JS::js_string(cell.sheet().interpreter().heap(), string);
}
diff --git a/Userland/Applications/Spreadsheet/CellType/String.h b/Userland/Applications/Spreadsheet/CellType/String.h
index 593bda55e2..15fcca374d 100644
--- a/Userland/Applications/Spreadsheet/CellType/String.h
+++ b/Userland/Applications/Spreadsheet/CellType/String.h
@@ -15,8 +15,8 @@ class StringCell : public CellType {
public:
StringCell();
virtual ~StringCell() override;
- virtual String display(Cell&, const CellTypeMetadata&) const override;
- virtual JS::Value js_value(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<String> display(Cell&, const CellTypeMetadata&) const override;
+ virtual JS::ThrowCompletionOr<JS::Value> js_value(Cell&, const CellTypeMetadata&) const override;
};
}
diff --git a/Userland/Applications/Spreadsheet/CellType/Type.h b/Userland/Applications/Spreadsheet/CellType/Type.h
index 025b14650d..89ab6ee11a 100644
--- a/Userland/Applications/Spreadsheet/CellType/Type.h
+++ b/Userland/Applications/Spreadsheet/CellType/Type.h
@@ -28,8 +28,8 @@ public:
static const CellType* get_by_name(StringView);
static Vector<StringView> names();
- virtual String display(Cell&, const CellTypeMetadata&) const = 0;
- virtual JS::Value js_value(Cell&, const CellTypeMetadata&) const = 0;
+ virtual JS::ThrowCompletionOr<String> display(Cell&, const CellTypeMetadata&) const = 0;
+ virtual JS::ThrowCompletionOr<JS::Value> js_value(Cell&, const CellTypeMetadata&) const = 0;
virtual ~CellType() { }
const String& name() const { return m_name; }
diff --git a/Userland/Applications/Spreadsheet/Spreadsheet.cpp b/Userland/Applications/Spreadsheet/Spreadsheet.cpp
index 83f8296c54..67d8143ba5 100644
--- a/Userland/Applications/Spreadsheet/Spreadsheet.cpp
+++ b/Userland/Applications/Spreadsheet/Spreadsheet.cpp
@@ -598,8 +598,11 @@ Vector<Vector<String>> Sheet::to_xsv() const
row.resize(column_count);
for (size_t j = 0; j < column_count; ++j) {
auto cell = at({ j, i });
- if (cell)
- row[j] = cell->typed_display();
+ if (cell) {
+ auto result = cell->typed_display();
+ if (result.has_value())
+ row[j] = result.value();
+ }
}
data.append(move(row));
diff --git a/Userland/Applications/Spreadsheet/SpreadsheetModel.cpp b/Userland/Applications/Spreadsheet/SpreadsheetModel.cpp
index 986b992148..86e78bf91f 100644
--- a/Userland/Applications/Spreadsheet/SpreadsheetModel.cpp
+++ b/Userland/Applications/Spreadsheet/SpreadsheetModel.cpp
@@ -27,29 +27,46 @@ GUI::Variant SheetModel::data(const GUI::ModelIndex& index, GUI::ModelRole role)
if (!cell)
return String::empty();
- if (cell->kind() == Spreadsheet::Cell::Formula) {
- if (auto exception = cell->exception()) {
- StringBuilder builder;
- builder.append("Error: ");
- auto value = exception->value();
- if (value.is_object()) {
- auto& object = value.as_object();
- if (is<JS::Error>(object)) {
- auto error = object.get_without_side_effects("message").to_string_without_side_effects();
- builder.append(error);
- return builder.to_string();
- }
+ Function<String(JS::Value)> to_string_as_exception = [&](JS::Value value) {
+ ScopeGuard clear_exception {
+ [&] {
+ cell->sheet().interpreter().vm().clear_exception();
+ }
+ };
+
+ StringBuilder builder;
+ builder.append("Error: "sv);
+ if (value.is_object()) {
+ auto& object = value.as_object();
+ if (is<JS::Error>(object)) {
+ auto message = object.get_without_side_effects("message");
+ auto error = message.to_string(cell->sheet().global_object());
+ if (error.is_throw_completion())
+ builder.append(message.to_string_without_side_effects());
+ else
+ builder.append(error.release_value());
+ return builder.to_string();
}
- auto error = value.to_string_without_side_effects();
- // This is annoying, but whatever.
- cell->sheet().interpreter().vm().clear_exception();
-
- builder.append(error);
- return builder.to_string();
}
+ auto error_message = value.to_string(cell->sheet().global_object());
+
+ if (error_message.is_throw_completion())
+ return to_string_as_exception(error_message.release_error().value());
+
+ builder.append(error_message.release_value());
+ return builder.to_string();
+ };
+
+ if (cell->kind() == Spreadsheet::Cell::Formula) {
+ if (auto exception = cell->exception())
+ return to_string_as_exception(exception->value());
}
- return cell->typed_display();
+ auto display = cell->typed_display();
+ if (display.is_error())
+ return to_string_as_exception(display.release_error().value());
+
+ return display.release_value();
}
if (role == GUI::ModelRole::MimeData)
@@ -155,5 +172,4 @@ void SheetModel::update()
m_sheet->update();
did_update(UpdateFlag::DontInvalidateIndices);
}
-
}