| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260 | /*************************************************************************//*  quick_open.cpp                                                       *//*************************************************************************//*                       This file is part of:                           *//*                           GODOT ENGINE                                *//*                      https://godotengine.org                          *//*************************************************************************//* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur.                 *//* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md).   *//*                                                                       *//* 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 "quick_open.h"#include "core/os/keyboard.h"void EditorQuickOpen::popup_dialog(const StringName &p_base, bool p_enable_multi, bool p_dontclear) {	base_type = p_base;	search_options->set_select_mode(p_enable_multi ? Tree::SELECT_MULTI : Tree::SELECT_SINGLE);	popup_centered_ratio(0.4);	if (p_dontclear) {		search_box->select_all();	} else {		search_box->clear();	}	search_box->grab_focus();	_update_search();}String EditorQuickOpen::get_selected() const {	TreeItem *ti = search_options->get_selected();	if (!ti) {		return String();	}	return "res://" + ti->get_text(0);}Vector<String> EditorQuickOpen::get_selected_files() const {	Vector<String> files;	TreeItem *item = search_options->get_next_selected(search_options->get_root());	while (item) {		files.push_back("res://" + item->get_text(0));		item = search_options->get_next_selected(item);	}	return files;}void EditorQuickOpen::_text_changed(const String &p_newtext) {	_update_search();}void EditorQuickOpen::_sbox_input(const Ref<InputEvent> &p_ie) {	Ref<InputEventKey> k = p_ie;	if (k.is_valid()) {		switch (k->get_keycode()) {			case KEY_UP:			case KEY_DOWN:			case KEY_PAGEUP:			case KEY_PAGEDOWN: {				search_options->call("_gui_input", k);				search_box->accept_event();				TreeItem *root = search_options->get_root();				if (!root->get_children()) {					break;				}				TreeItem *current = search_options->get_selected();				TreeItem *item = search_options->get_next_selected(root);				while (item) {					item->deselect(0);					item = search_options->get_next_selected(item);				}				current->select(0);				current->set_as_cursor(0);			} break;		}	}}float EditorQuickOpen::_score_path(String search, String path) const {	// Positive bias for matches close to the _beginning of the file name_.	String file = path.get_file();	int pos = file.findn(search);	if (pos != -1) {		return 1.0f - 0.1f * (float(pos) / file.length());	}	// Positive bias for matches close to the _end of the path_.	String base = path.get_base_dir();	pos = base.rfindn(search);	if (pos != -1) {		return 0.9f - 0.1f * (float(base.length() - pos) / base.length());	}	// Results that contain all characters but not the string.	return path.similarity(search) * 0.8f;}void EditorQuickOpen::_parse_fs(EditorFileSystemDirectory *efsd, Vector<Pair<String, Ref<Texture2D>>> &list) {	for (int i = 0; i < efsd->get_subdir_count(); i++) {		_parse_fs(efsd->get_subdir(i), list);	}	for (int i = 0; i < efsd->get_file_count(); i++) {		StringName file_type = efsd->get_file_type(i);		if (ClassDB::is_parent_class(file_type, base_type)) {			String file = efsd->get_file_path(i);			file = file.substr(6, file.length());			if (search_box->get_text().is_subsequence_ofi(file)) {				Pair<String, Ref<Texture2D>> pair;				pair.first = file;				pair.second = search_options->get_theme_icon(search_options->has_theme_icon(file_type, ei) ? file_type : ot, ei);				list.push_back(pair);			}		}	}}Vector<Pair<String, Ref<Texture2D>>> EditorQuickOpen::_sort_fs(Vector<Pair<String, Ref<Texture2D>>> &list) {	String search_text = search_box->get_text().to_lower();	Vector<Pair<String, Ref<Texture2D>>> sorted_list;	if (search_text == String() || list.size() == 0) {		return list;	}	Vector<float> scores;	scores.resize(list.size());	for (int i = 0; i < list.size(); i++) {		scores.write[i] = _score_path(search_text, list[i].first.to_lower());	}	while (list.size() > 0) {		float best_score = 0.0f;		int best_idx = 0;		for (int i = 0; i < list.size(); i++) {			float current_score = scores[i];			if (current_score > best_score) {				best_score = current_score;				best_idx = i;			}		}		sorted_list.push_back(list[best_idx]);		list.remove(best_idx);		scores.remove(best_idx);	}	return sorted_list;}void EditorQuickOpen::_update_search() {	search_options->clear();	TreeItem *root = search_options->create_item();	EditorFileSystemDirectory *efsd = EditorFileSystem::get_singleton()->get_filesystem();	Vector<Pair<String, Ref<Texture2D>>> list;	_parse_fs(efsd, list);	list = _sort_fs(list);	for (int i = 0; i < list.size(); i++) {		TreeItem *ti = search_options->create_item(root);		ti->set_text(0, list[i].first);		ti->set_icon(0, list[i].second);	}	TreeItem *result = root->get_children();	if (result) {		result->select(0);		result->set_as_cursor(0);	}	get_ok()->set_disabled(!result);}void EditorQuickOpen::_confirmed() {	if (!search_options->get_selected()) {		return;	}	emit_signal("quick_open");	hide();}void EditorQuickOpen::_theme_changed() {	search_box->set_right_icon(search_options->get_theme_icon("Search", ei));}void EditorQuickOpen::_notification(int p_what) {	switch (p_what) {		case NOTIFICATION_ENTER_TREE: {			connect("confirmed", callable_mp(this, &EditorQuickOpen::_confirmed));			search_box->set_clear_button_enabled(true);		} break;		case NOTIFICATION_EXIT_TREE: {			disconnect("confirmed", callable_mp(this, &EditorQuickOpen::_confirmed));		} break;	}}StringName EditorQuickOpen::get_base_type() const {	return base_type;}void EditorQuickOpen::_bind_methods() {	ADD_SIGNAL(MethodInfo("quick_open"));}EditorQuickOpen::EditorQuickOpen() {	VBoxContainer *vbc = memnew(VBoxContainer);	vbc->connect("theme_changed", callable_mp(this, &EditorQuickOpen::_theme_changed));	add_child(vbc);	search_box = memnew(LineEdit);	search_box->connect("text_changed", callable_mp(this, &EditorQuickOpen::_text_changed));	search_box->connect("gui_input", callable_mp(this, &EditorQuickOpen::_sbox_input));	vbc->add_margin_child(TTR("Search:"), search_box);	search_options = memnew(Tree);	search_options->connect("item_activated", callable_mp(this, &EditorQuickOpen::_confirmed));	search_options->set_hide_root(true);	search_options->set_hide_folding(true);	search_options->add_theme_constant_override("draw_guides", 1);	vbc->add_margin_child(TTR("Matches:"), search_options, true);	get_ok()->set_text(TTR("Open"));	register_text_enter(search_box);	set_hide_on_ok(false);	ei = "EditorIcons";	ot = "Object";}
 |