mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-12-04 05:20:30 +00:00
221 lines
6.2 KiB
C++
221 lines
6.2 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "RemoteObjectPropertyModel.h"
|
|
#include "RemoteObject.h"
|
|
#include "RemoteProcess.h"
|
|
|
|
namespace Inspector {
|
|
|
|
RemoteObjectPropertyModel::RemoteObjectPropertyModel(RemoteObject& object)
|
|
: m_object(object)
|
|
{
|
|
}
|
|
|
|
int RemoteObjectPropertyModel::row_count(const GUI::ModelIndex& index) const
|
|
{
|
|
Function<int(const JsonValue&)> do_count = [&](const JsonValue& value) {
|
|
if (value.is_array())
|
|
return value.as_array().size();
|
|
else if (value.is_object())
|
|
return value.as_object().size();
|
|
return (size_t)0;
|
|
};
|
|
|
|
if (index.is_valid()) {
|
|
auto* path = static_cast<const JsonPath*>(index.internal_data());
|
|
return do_count(path->resolve(m_object.json));
|
|
} else {
|
|
return do_count(m_object.json);
|
|
}
|
|
}
|
|
|
|
String RemoteObjectPropertyModel::column_name(int column) const
|
|
{
|
|
switch (column) {
|
|
case Column::Name:
|
|
return "Name";
|
|
case Column::Value:
|
|
return "Value";
|
|
}
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
|
|
GUI::Variant RemoteObjectPropertyModel::data(const GUI::ModelIndex& index, GUI::ModelRole role) const
|
|
{
|
|
auto* path = static_cast<const JsonPath*>(index.internal_data());
|
|
if (!path)
|
|
return {};
|
|
|
|
if (role == GUI::ModelRole::Display) {
|
|
switch (index.column()) {
|
|
case Column::Name:
|
|
return path->last().to_string();
|
|
case Column::Value: {
|
|
auto data = path->resolve(m_object.json);
|
|
if (data.is_array())
|
|
return String::formatted("<Array with {} element{}", data.as_array().size(), data.as_array().size() == 1 ? ">" : "s>");
|
|
if (data.is_object())
|
|
return String::formatted("<Object with {} entr{}", data.as_object().size(), data.as_object().size() == 1 ? "y>" : "ies>");
|
|
return data;
|
|
}
|
|
}
|
|
}
|
|
return {};
|
|
}
|
|
|
|
void RemoteObjectPropertyModel::update()
|
|
{
|
|
did_update();
|
|
}
|
|
|
|
void RemoteObjectPropertyModel::set_data(const GUI::ModelIndex& index, const GUI::Variant& new_value)
|
|
{
|
|
if (!index.is_valid())
|
|
return;
|
|
|
|
auto* path = static_cast<const JsonPath*>(index.internal_data());
|
|
if (path->size() != 1)
|
|
return;
|
|
|
|
FlatPtr address = m_object.address;
|
|
RemoteProcess::the().set_property(address, path->first().to_string(), new_value.to_string());
|
|
did_update();
|
|
}
|
|
|
|
GUI::ModelIndex RemoteObjectPropertyModel::index(int row, int column, const GUI::ModelIndex& parent) const
|
|
{
|
|
const auto& parent_path = parent.is_valid() ? *static_cast<const JsonPath*>(parent.internal_data()) : JsonPath {};
|
|
|
|
auto nth_child = [&](int n, const JsonValue& value) -> const JsonPath* {
|
|
auto path = make<JsonPath>();
|
|
path->extend(parent_path);
|
|
int row_index = n;
|
|
if (value.is_object()) {
|
|
String property_name;
|
|
auto& object = value.as_object();
|
|
object.for_each_member([&](auto& name, auto&) {
|
|
if (row_index > 0) {
|
|
--row_index;
|
|
} else if (row_index == 0) {
|
|
property_name = name;
|
|
--row_index;
|
|
}
|
|
});
|
|
if (property_name.is_null())
|
|
return nullptr;
|
|
|
|
path->append({ property_name });
|
|
m_paths.append(move(path));
|
|
} else if (value.is_array()) {
|
|
path->append(JsonPathElement { (size_t)n });
|
|
m_paths.append(move(path));
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
return &m_paths.last();
|
|
};
|
|
|
|
if (!parent.is_valid()) {
|
|
if (m_object.json.is_empty())
|
|
return {};
|
|
}
|
|
|
|
auto index_path = cached_path_at(row, parent_path);
|
|
|
|
if (!index_path)
|
|
index_path = nth_child(row, parent_path.resolve(m_object.json));
|
|
|
|
if (!index_path)
|
|
return {};
|
|
|
|
return create_index(row, column, index_path);
|
|
}
|
|
|
|
GUI::ModelIndex RemoteObjectPropertyModel::parent_index(const GUI::ModelIndex& index) const
|
|
{
|
|
if (!index.is_valid())
|
|
return index;
|
|
|
|
auto path = *static_cast<const JsonPath*>(index.internal_data());
|
|
if (path.is_empty())
|
|
return {};
|
|
|
|
path.take_last();
|
|
if (path.is_empty())
|
|
return {};
|
|
|
|
auto* cpath = find_cached_path(path);
|
|
if (cpath) {
|
|
int index_in_parent = 0;
|
|
if (cpath->last().kind() == JsonPathElement::Kind::Index)
|
|
index_in_parent = cpath->last().index();
|
|
else if (cpath->last().kind() == JsonPathElement::Kind::Key) {
|
|
auto path_copy = path;
|
|
auto last = path_copy.take_last();
|
|
bool found = false;
|
|
path_copy.resolve(m_object.json).as_object().for_each_member([&](auto& name, auto&) {
|
|
if (!found) {
|
|
if (last.key() == name)
|
|
found = true;
|
|
else
|
|
index_in_parent++;
|
|
}
|
|
});
|
|
}
|
|
return create_index(index_in_parent, 0, cpath);
|
|
}
|
|
|
|
dbgln("No cached path found for path {}", path.to_string());
|
|
return {};
|
|
}
|
|
|
|
const JsonPath* RemoteObjectPropertyModel::cached_path_at(int n, const Vector<JsonPathElement>& prefix) const
|
|
{
|
|
// FIXME: ModelIndex wants a void*, so we have to keep these
|
|
// indices alive, but allocating a new path every time
|
|
// we're asked for an index is silly, so we have to look for existing ones first.
|
|
const JsonPath* index_path = nullptr;
|
|
int row_index = n;
|
|
for (auto& path : m_paths) {
|
|
if (path.size() != prefix.size() + 1)
|
|
continue;
|
|
|
|
for (size_t i = 0; i < prefix.size(); ++i) {
|
|
if (path[i] != prefix[i])
|
|
goto do_continue;
|
|
}
|
|
|
|
if (row_index == 0) {
|
|
index_path = &path;
|
|
break;
|
|
}
|
|
--row_index;
|
|
do_continue:;
|
|
}
|
|
|
|
return index_path;
|
|
};
|
|
|
|
const JsonPath* RemoteObjectPropertyModel::find_cached_path(const Vector<JsonPathElement>& path) const
|
|
{
|
|
for (auto& cpath : m_paths) {
|
|
if (cpath.size() != path.size())
|
|
continue;
|
|
|
|
for (size_t i = 0; i < cpath.size(); ++i) {
|
|
if (cpath[i] != path[i])
|
|
goto do_continue;
|
|
}
|
|
|
|
return &cpath;
|
|
do_continue:;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
}
|