/*************************************************************************/ /* create_dialog.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* http://www.godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2017 Juan Linietsky, Ariel Manzur. */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /*************************************************************************/ #include "create_dialog.h" #include "object_type_db.h" #include "print_string.h" #include "scene/gui/box_container.h" #include "editor_node.h" #if 1 #include "os/keyboard.h" #include "editor_settings.h" #include "editor_help.h" void CreateDialog::popup(bool p_dontclear) { recent->clear(); FileAccess *f = FileAccess::open( EditorSettings::get_singleton()->get_project_settings_path().plus_file("create_recent."+base_type), FileAccess::READ ); if (f) { TreeItem *root = recent->create_item(); while(!f->eof_reached()) { String l = f->get_line().strip_edges(); if (l!=String()) { TreeItem *ti = recent->create_item(root); ti->set_text(0,l); if (has_icon(l,"EditorIcons")) { ti->set_icon(0,get_icon(l,"EditorIcons")); } else { ti->set_icon(0,get_icon("Object","EditorIcons")); } } } memdelete(f); } favorites->clear(); f = FileAccess::open( EditorSettings::get_singleton()->get_project_settings_path().plus_file("favorites."+base_type), FileAccess::READ ); favorite_list.clear(); if (f) { while(!f->eof_reached()) { String l = f->get_line().strip_edges(); if (l!=String()) { favorite_list.push_back(l); } } memdelete(f); } else { #if 0 // I think this was way too confusing if (base_type=="Node") { //harcode some favorites :D favorite_list.push_back("Panel"); favorite_list.push_back("Button"); favorite_list.push_back("Label"); favorite_list.push_back("LineEdit"); favorite_list.push_back("Node2D"); favorite_list.push_back("Sprite"); favorite_list.push_back("Camera2D"); favorite_list.push_back("Area2D"); favorite_list.push_back("CollisionShape2D"); favorite_list.push_back("Spatial"); favorite_list.push_back("Camera"); favorite_list.push_back("Area"); favorite_list.push_back("CollisionShape"); favorite_list.push_back("TestCube"); favorite_list.push_back("AnimationPlayer"); } #endif } _update_favorite_list(); popup_centered_ratio(); if (p_dontclear) search_box->select_all(); else { search_box->clear(); } search_box->grab_focus(); _update_search(); } void CreateDialog::_text_changed(const String& p_newtext) { _update_search(); } void CreateDialog::_sbox_input(const InputEvent& p_ie) { if (p_ie.type==InputEvent::KEY && ( p_ie.key.scancode == KEY_UP || p_ie.key.scancode == KEY_DOWN || p_ie.key.scancode == KEY_PAGEUP || p_ie.key.scancode == KEY_PAGEDOWN ) ) { search_options->call("_gui_input",p_ie); search_box->accept_event(); } } void CreateDialog::add_type(const String& p_type,HashMap& p_types,TreeItem *p_root,TreeItem **to_select) { if (p_types.has(p_type)) return; if (!ClassDB::is_parent_class(p_type,base_type) || p_type==base_type) return; String inherits=ClassDB::get_parent_class(p_type); TreeItem *parent=p_root; if (inherits.length()) { if (!p_types.has(inherits)) { add_type(inherits,p_types,p_root,to_select); } if (p_types.has(inherits) ) parent=p_types[inherits]; } TreeItem *item = search_options->create_item(parent); item->set_text(0,p_type); if (!ClassDB::can_instance(p_type)) { item->set_custom_color(0, Color(0.5,0.5,0.5) ); item->set_selectable(0,false); } else { if ((!*to_select && (search_box->get_text().is_subsequence_ofi(p_type))) || search_box->get_text()==p_type) { *to_select=item; } } if (bool(EditorSettings::get_singleton()->get("docks/scene_tree/start_create_dialog_fully_expanded"))) { item->set_collapsed(false); } else { // don't collapse search results bool collapse = (search_box->get_text() == ""); // don't collapse the root node collapse &= (item != p_root); // don't collapse abstract nodes on the first tree level collapse &= ((parent != p_root) || (ClassDB::can_instance(p_type))); item->set_collapsed(collapse); } const String& description = EditorHelp::get_doc_data()->class_list[p_type].brief_description; item->set_tooltip(0,description); if (has_icon(p_type,"EditorIcons")) { item->set_icon(0, get_icon(p_type,"EditorIcons")); } p_types[p_type]=item; } void CreateDialog::_update_search() { search_options->clear(); favorite->set_disabled(true); help_bit->set_text(""); /* TreeItem *root = search_options->create_item(); _parse_fs(EditorFileSystem::get_singleton()->get_filesystem()); */ List type_list; ClassDB::get_class_list(&type_list); HashMap types; TreeItem *root = search_options->create_item(); root->set_text(0,base_type); if (has_icon(base_type,"EditorIcons")) { root->set_icon(0,get_icon(base_type,"EditorIcons")); } List::Element *I=type_list.front(); TreeItem *to_select=NULL; for(;I;I=I->next()) { String type=I->get(); if (base_type=="Node" && type.begins_with("Editor")) continue; // do not show editor nodes if (!ClassDB::can_instance(type)) continue; // cant create what can't be instanced if (search_box->get_text()=="") { add_type(type,types,root,&to_select); } else { bool found=false; String type=I->get(); while(type!="" && ClassDB::is_parent_class(type,base_type) && type!=base_type) { if (search_box->get_text().is_subsequence_ofi(type)) { found=true; break; } type=ClassDB::get_parent_class(type); } if (found) add_type(I->get(),types,root,&to_select); } if (EditorNode::get_editor_data().get_custom_types().has(type) && ClassDB::is_parent_class(type, base_type)) { //there are custom types based on this... cool. //print_line("there are custom types"); const Vector &ct = EditorNode::get_editor_data().get_custom_types()[type]; for(int i=0;iget_text().is_subsequence_ofi(ct[i].name); if (!show) continue; if (!types.has(type)) add_type(type,types,root,&to_select); TreeItem *ti; if (types.has(type) ) ti=types[type]; else ti=search_options->get_root(); TreeItem *item = search_options->create_item(ti); item->set_metadata(0,type); item->set_text(0,ct[i].name); if (ct[i].icon.is_valid()) { item->set_icon(0,ct[i].icon); } if (!to_select || ct[i].name==search_box->get_text()) { to_select=item; } } } } if (to_select) { to_select->select(0); favorite->set_disabled(false); favorite->set_pressed(favorite_list.find(to_select->get_text(0))!=-1); } get_ok()->set_disabled(root->get_children()==NULL); } void CreateDialog::_confirmed() { TreeItem *ti = search_options->get_selected(); if (!ti) return; FileAccess *f = FileAccess::open( EditorSettings::get_singleton()->get_project_settings_path().plus_file("create_recent."+base_type), FileAccess::WRITE ); if (f) { f->store_line(get_selected_type()); TreeItem *t = recent->get_root(); if (t) t=t->get_children(); int count=0; while(t) { if (t->get_text(0)!=get_selected_type()) { f->store_line(t->get_text(0)); } if (count>32) { //limit it to 32 entries.. break; } t=t->get_next(); count++; } memdelete(f); } emit_signal("create"); hide(); } void CreateDialog::_notification(int p_what) { if (p_what==NOTIFICATION_ENTER_TREE) { connect("confirmed",this,"_confirmed"); favorite->set_icon(get_icon("Favorites","EditorIcons")); } if (p_what==NOTIFICATION_EXIT_TREE) { disconnect("confirmed",this,"_confirmed"); } if (p_what==NOTIFICATION_VISIBILITY_CHANGED) { if (is_visible()) { search_box->call_deferred("grab_focus"); // still not visible search_box->select_all(); } } } void CreateDialog::set_base_type(const String& p_base) { base_type=p_base; set_title(TTR("Create New")+" "+p_base); _update_search(); } String CreateDialog::get_selected_type() { TreeItem *selected = search_options->get_selected(); if (selected) return selected->get_text(0); else return String(); } Object *CreateDialog::instance_selected() { TreeItem *selected = search_options->get_selected(); if (selected) { Variant md = selected->get_metadata(0); String custom; if (md.get_type()!=Variant::NIL) custom=md; if (custom!=String()) { if (EditorNode::get_editor_data().get_custom_types().has(custom)) { for(int i=0;iget_text(0)) { Ref icon = EditorNode::get_editor_data().get_custom_types()[custom][i].icon; Ref