Merge pull request #59680 from fire-forge/property-help
Add "Open Documentation" button to EditorProperty context menu
This commit is contained in:
commit
9b29f18631
3 changed files with 47 additions and 23 deletions
|
@ -39,6 +39,7 @@
|
||||||
#include "editor/editor_property_name_processor.h"
|
#include "editor/editor_property_name_processor.h"
|
||||||
#include "editor/editor_scale.h"
|
#include "editor/editor_scale.h"
|
||||||
#include "editor/editor_settings.h"
|
#include "editor/editor_settings.h"
|
||||||
|
#include "editor/plugins/script_editor_plugin.h"
|
||||||
#include "multi_node_edit.h"
|
#include "multi_node_edit.h"
|
||||||
#include "scene/gui/center_container.h"
|
#include "scene/gui/center_container.h"
|
||||||
#include "scene/property_utils.h"
|
#include "scene/property_utils.h"
|
||||||
|
@ -410,6 +411,10 @@ StringName EditorProperty::get_edited_property() const {
|
||||||
return property;
|
return property;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EditorProperty::set_doc_path(const String &p_doc_path) {
|
||||||
|
doc_path = p_doc_path;
|
||||||
|
}
|
||||||
|
|
||||||
void EditorProperty::update_property() {
|
void EditorProperty::update_property() {
|
||||||
GDVIRTUAL_CALL(_update_property);
|
GDVIRTUAL_CALL(_update_property);
|
||||||
}
|
}
|
||||||
|
@ -906,6 +911,10 @@ void EditorProperty::menu_option(int p_option) {
|
||||||
emit_signal(SNAME("property_pinned"), property, !pinned);
|
emit_signal(SNAME("property_pinned"), property, !pinned);
|
||||||
update();
|
update();
|
||||||
} break;
|
} break;
|
||||||
|
case MENU_OPEN_DOCUMENTATION: {
|
||||||
|
ScriptEditor::get_singleton()->goto_help(doc_path);
|
||||||
|
EditorNode::get_singleton()->set_visible_editor(EditorNode::EDITOR_SCRIPT);
|
||||||
|
} break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -985,20 +994,25 @@ void EditorProperty::_update_popup() {
|
||||||
add_child(menu);
|
add_child(menu);
|
||||||
menu->connect("id_pressed", callable_mp(this, &EditorProperty::menu_option));
|
menu->connect("id_pressed", callable_mp(this, &EditorProperty::menu_option));
|
||||||
}
|
}
|
||||||
menu->add_shortcut(ED_GET_SHORTCUT("property_editor/copy_property"), MENU_COPY_PROPERTY);
|
menu->add_icon_shortcut(get_theme_icon(SNAME("ActionCopy"), SNAME("EditorIcons")), ED_GET_SHORTCUT("property_editor/copy_property"), MENU_COPY_PROPERTY);
|
||||||
menu->add_shortcut(ED_GET_SHORTCUT("property_editor/paste_property"), MENU_PASTE_PROPERTY);
|
menu->add_icon_shortcut(get_theme_icon(SNAME("ActionPaste"), SNAME("EditorIcons")), ED_GET_SHORTCUT("property_editor/paste_property"), MENU_PASTE_PROPERTY);
|
||||||
menu->add_shortcut(ED_GET_SHORTCUT("property_editor/copy_property_path"), MENU_COPY_PROPERTY_PATH);
|
menu->add_icon_shortcut(get_theme_icon(SNAME("CopyNodePath"), SNAME("EditorIcons")), ED_GET_SHORTCUT("property_editor/copy_property_path"), MENU_COPY_PROPERTY_PATH);
|
||||||
menu->set_item_disabled(MENU_PASTE_PROPERTY, is_read_only());
|
menu->set_item_disabled(MENU_PASTE_PROPERTY, is_read_only());
|
||||||
if (!pin_hidden) {
|
if (!pin_hidden) {
|
||||||
menu->add_separator();
|
menu->add_separator();
|
||||||
if (can_pin) {
|
if (can_pin) {
|
||||||
menu->add_check_item(TTR("Pin value"), MENU_PIN_VALUE);
|
menu->add_icon_check_item(get_theme_icon(SNAME("Pin"), SNAME("EditorIcons")), TTR("Pin Value"), MENU_PIN_VALUE);
|
||||||
menu->set_item_checked(menu->get_item_index(MENU_PIN_VALUE), pinned);
|
menu->set_item_checked(menu->get_item_index(MENU_PIN_VALUE), pinned);
|
||||||
menu->set_item_tooltip(menu->get_item_index(MENU_PIN_VALUE), TTR("Pinning a value forces it to be saved even if it's equal to the default."));
|
|
||||||
} else {
|
} else {
|
||||||
menu->add_check_item(vformat(TTR("Pin value [Disabled because '%s' is editor-only]"), property), MENU_PIN_VALUE);
|
menu->add_icon_check_item(get_theme_icon(SNAME("Pin"), SNAME("EditorIcons")), vformat(TTR("Pin Value [Disabled because '%s' is editor-only]"), property), MENU_PIN_VALUE);
|
||||||
menu->set_item_disabled(menu->get_item_index(MENU_PIN_VALUE), true);
|
menu->set_item_disabled(menu->get_item_index(MENU_PIN_VALUE), true);
|
||||||
}
|
}
|
||||||
|
menu->set_item_tooltip(menu->get_item_index(MENU_PIN_VALUE), TTR("Pinning a value forces it to be saved even if it's equal to the default."));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!doc_path.is_empty()) {
|
||||||
|
menu->add_separator();
|
||||||
|
menu->add_icon_item(get_theme_icon(SNAME("Help"), SNAME("EditorIcons")), TTR("Open Documentation"), MENU_OPEN_DOCUMENTATION);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2844,7 +2858,7 @@ void EditorInspector::update_tree() {
|
||||||
restart_request_props.insert(p.name);
|
restart_request_props.insert(p.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
String doc_hint;
|
PropertyDocInfo doc_info;
|
||||||
|
|
||||||
if (use_doc_hints) {
|
if (use_doc_hints) {
|
||||||
// Build the doc hint, to use as tooltip.
|
// Build the doc hint, to use as tooltip.
|
||||||
|
@ -2856,16 +2870,15 @@ void EditorInspector::update_tree() {
|
||||||
}
|
}
|
||||||
|
|
||||||
StringName propname = property_prefix + p.name;
|
StringName propname = property_prefix + p.name;
|
||||||
String descr;
|
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
// Search for the property description in the cache.
|
// Search for the property description in the cache.
|
||||||
HashMap<StringName, HashMap<StringName, String>>::Iterator E = descr_cache.find(classname);
|
HashMap<StringName, HashMap<StringName, PropertyDocInfo>>::Iterator E = doc_info_cache.find(classname);
|
||||||
if (E) {
|
if (E) {
|
||||||
HashMap<StringName, String>::Iterator F = E->value.find(propname);
|
HashMap<StringName, PropertyDocInfo>::Iterator F = E->value.find(propname);
|
||||||
if (F) {
|
if (F) {
|
||||||
found = true;
|
found = true;
|
||||||
descr = F->value;
|
doc_info = F->value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2873,10 +2886,11 @@ void EditorInspector::update_tree() {
|
||||||
// Build the property description String and add it to the cache.
|
// Build the property description String and add it to the cache.
|
||||||
DocTools *dd = EditorHelp::get_doc_data();
|
DocTools *dd = EditorHelp::get_doc_data();
|
||||||
HashMap<String, DocData::ClassDoc>::Iterator F = dd->class_list.find(classname);
|
HashMap<String, DocData::ClassDoc>::Iterator F = dd->class_list.find(classname);
|
||||||
while (F && descr.is_empty()) {
|
while (F && doc_info.description.is_empty()) {
|
||||||
for (int i = 0; i < F->value.properties.size(); i++) {
|
for (int i = 0; i < F->value.properties.size(); i++) {
|
||||||
if (F->value.properties[i].name == propname.operator String()) {
|
if (F->value.properties[i].name == propname.operator String()) {
|
||||||
descr = DTR(F->value.properties[i].description);
|
doc_info.description = DTR(F->value.properties[i].description);
|
||||||
|
doc_info.path = "class_property:" + F->value.name + ":" + F->value.properties[i].name;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2885,7 +2899,8 @@ void EditorInspector::update_tree() {
|
||||||
if (slices.size() == 2 && slices[0].begins_with("theme_override_")) {
|
if (slices.size() == 2 && slices[0].begins_with("theme_override_")) {
|
||||||
for (int i = 0; i < F->value.theme_properties.size(); i++) {
|
for (int i = 0; i < F->value.theme_properties.size(); i++) {
|
||||||
if (F->value.theme_properties[i].name == slices[1]) {
|
if (F->value.theme_properties[i].name == slices[1]) {
|
||||||
descr = DTR(F->value.theme_properties[i].description);
|
doc_info.description = DTR(F->value.theme_properties[i].description);
|
||||||
|
doc_info.path = "class_theme_item:" + F->value.name + ":" + F->value.theme_properties[i].name;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2897,10 +2912,9 @@ void EditorInspector::update_tree() {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
descr_cache[classname][propname] = descr;
|
|
||||||
}
|
|
||||||
|
|
||||||
doc_hint = descr;
|
doc_info_cache[classname][propname] = doc_info;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<EditorInspectorPlugin::AddedEditor> editors;
|
Vector<EditorInspectorPlugin::AddedEditor> editors;
|
||||||
|
@ -2983,11 +2997,12 @@ void EditorInspector::update_tree() {
|
||||||
ep->connect("multiple_properties_changed", callable_mp(this, &EditorInspector::_multiple_properties_changed));
|
ep->connect("multiple_properties_changed", callable_mp(this, &EditorInspector::_multiple_properties_changed));
|
||||||
ep->connect("resource_selected", callable_mp(this, &EditorInspector::_resource_selected), varray(), CONNECT_DEFERRED);
|
ep->connect("resource_selected", callable_mp(this, &EditorInspector::_resource_selected), varray(), CONNECT_DEFERRED);
|
||||||
ep->connect("object_id_selected", callable_mp(this, &EditorInspector::_object_id_selected), varray(), CONNECT_DEFERRED);
|
ep->connect("object_id_selected", callable_mp(this, &EditorInspector::_object_id_selected), varray(), CONNECT_DEFERRED);
|
||||||
if (!doc_hint.is_empty()) {
|
if (!doc_info.description.is_empty()) {
|
||||||
ep->set_tooltip(property_prefix + p.name + "::" + doc_hint);
|
ep->set_tooltip(property_prefix + p.name + "::" + doc_info.description);
|
||||||
} else {
|
} else {
|
||||||
ep->set_tooltip(property_prefix + p.name);
|
ep->set_tooltip(property_prefix + p.name);
|
||||||
}
|
}
|
||||||
|
ep->set_doc_path(doc_info.path);
|
||||||
ep->update_property();
|
ep->update_property();
|
||||||
ep->_update_pin_flags();
|
ep->_update_pin_flags();
|
||||||
ep->update_revert_and_pin_status();
|
ep->update_revert_and_pin_status();
|
||||||
|
@ -3473,7 +3488,7 @@ void EditorInspector::_property_pinned(const String &p_path, bool p_pinned) {
|
||||||
void EditorInspector::_property_selected(const String &p_path, int p_focusable) {
|
void EditorInspector::_property_selected(const String &p_path, int p_focusable) {
|
||||||
property_selected = p_path;
|
property_selected = p_path;
|
||||||
property_focusable = p_focusable;
|
property_focusable = p_focusable;
|
||||||
//deselect the others
|
// Deselect the others.
|
||||||
for (const KeyValue<StringName, List<EditorProperty *>> &F : editor_property_map) {
|
for (const KeyValue<StringName, List<EditorProperty *>> &F : editor_property_map) {
|
||||||
if (F.key == property_selected) {
|
if (F.key == property_selected) {
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -62,6 +62,7 @@ public:
|
||||||
MENU_PASTE_PROPERTY,
|
MENU_PASTE_PROPERTY,
|
||||||
MENU_COPY_PROPERTY_PATH,
|
MENU_COPY_PROPERTY_PATH,
|
||||||
MENU_PIN_VALUE,
|
MENU_PIN_VALUE,
|
||||||
|
MENU_OPEN_DOCUMENTATION,
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -71,6 +72,7 @@ private:
|
||||||
Object *object = nullptr;
|
Object *object = nullptr;
|
||||||
StringName property;
|
StringName property;
|
||||||
String property_path;
|
String property_path;
|
||||||
|
String doc_path;
|
||||||
|
|
||||||
int property_usage;
|
int property_usage;
|
||||||
|
|
||||||
|
@ -148,6 +150,8 @@ public:
|
||||||
Object *get_edited_object();
|
Object *get_edited_object();
|
||||||
StringName get_edited_property() const;
|
StringName get_edited_property() const;
|
||||||
|
|
||||||
|
void set_doc_path(const String &p_doc_path);
|
||||||
|
|
||||||
virtual void update_property();
|
virtual void update_property();
|
||||||
void update_revert_and_pin_status();
|
void update_revert_and_pin_status();
|
||||||
|
|
||||||
|
@ -439,7 +443,7 @@ class EditorInspector : public ScrollContainer {
|
||||||
|
|
||||||
VBoxContainer *main_vbox = nullptr;
|
VBoxContainer *main_vbox = nullptr;
|
||||||
|
|
||||||
//map use to cache the instantiated editors
|
// Map used to cache the instantiated editors.
|
||||||
HashMap<StringName, List<EditorProperty *>> editor_property_map;
|
HashMap<StringName, List<EditorProperty *>> editor_property_map;
|
||||||
List<EditorInspectorSection *> sections;
|
List<EditorInspectorSection *> sections;
|
||||||
HashSet<StringName> pending;
|
HashSet<StringName> pending;
|
||||||
|
@ -473,7 +477,12 @@ class EditorInspector : public ScrollContainer {
|
||||||
int property_focusable;
|
int property_focusable;
|
||||||
int update_scroll_request;
|
int update_scroll_request;
|
||||||
|
|
||||||
HashMap<StringName, HashMap<StringName, String>> descr_cache;
|
struct PropertyDocInfo {
|
||||||
|
String description;
|
||||||
|
String path;
|
||||||
|
};
|
||||||
|
|
||||||
|
HashMap<StringName, HashMap<StringName, PropertyDocInfo>> doc_info_cache;
|
||||||
HashMap<StringName, String> class_descr_cache;
|
HashMap<StringName, String> class_descr_cache;
|
||||||
HashSet<StringName> restart_request_props;
|
HashSet<StringName> restart_request_props;
|
||||||
|
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<svg height="16" viewBox="0 0 16 16" width="16" xmlns="http://www.w3.org/2000/svg"><g fill="#e0e0e0" transform="translate(0 -1036.4)"><circle cx="3" cy="1048.4"/><path d="m2 1c-.55226.0001-.99994.4477-1 1v12c.0000552.5523.44774.9999 1 1h12c.55226-.0001.99994-.4477 1-1v-8l-5-5zm1 2h6v3c0 .554.44599 1 1 1h3v6h-10zm3 5-2 4h2l2-4zm4 0-2 4h2l2-4z" fill-opacity=".78431" transform="translate(0 1036.4)"/></g></svg>
|
<svg height="16" viewBox="0 0 16 16" width="16" xmlns="http://www.w3.org/2000/svg"><g fill="#e0e0e0" transform="translate(0 -1036.4)"><circle cx="3" cy="1048.4"/><path d="m2 1c-.55226.0001-.99994.4477-1 1v12c.0000552.5523.44774.9999 1 1h12c.55226-.0001.99994-.4477 1-1v-8l-5-5zm1 2h6v3c0 .554.44599 1 1 1h3v6h-10zm3 5-2 4h2l2-4zm4 0-2 4h2l2-4z" transform="translate(0 1036.4)"/></g></svg>
|
||||||
|
|
Before Width: | Height: | Size: 411 B After Width: | Height: | Size: 389 B |
Loading…
Reference in a new issue