2020-04-03 08:01:23 +00:00
|
|
|
#include "Search.hpp"
|
2020-03-13 21:37:18 +00:00
|
|
|
|
|
|
|
#include <cstddef>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <numeric>
|
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
#include <regex>
|
|
|
|
#include <future>
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <boost/optional.hpp>
|
|
|
|
#include <boost/filesystem/path.hpp>
|
|
|
|
#include <boost/filesystem/operations.hpp>
|
|
|
|
#include <boost/log/trivial.hpp>
|
|
|
|
|
2020-04-03 08:01:23 +00:00
|
|
|
//#include <wx/bmpcbox.h>
|
2020-03-13 21:37:18 +00:00
|
|
|
#include "libslic3r/PrintConfig.hpp"
|
|
|
|
#include "GUI_App.hpp"
|
|
|
|
#include "Tab.hpp"
|
|
|
|
#include "PresetBundle.hpp"
|
|
|
|
|
2020-03-25 09:24:43 +00:00
|
|
|
#define FTS_FUZZY_MATCH_IMPLEMENTATION
|
|
|
|
#include "fts_fuzzy_match.h"
|
|
|
|
|
2020-03-31 20:46:12 +00:00
|
|
|
#include "imgui/imconfig.h"
|
|
|
|
|
2020-03-13 21:37:18 +00:00
|
|
|
using boost::optional;
|
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
2020-03-31 20:46:12 +00:00
|
|
|
bool SearchOptions::Option::fuzzy_match_simple(char const * search_pattern) const
|
2020-03-13 21:37:18 +00:00
|
|
|
{
|
2020-03-25 09:24:43 +00:00
|
|
|
char const* opt_key_str = opt_key.c_str();
|
|
|
|
char const* label_str = label.utf8_str();
|
|
|
|
|
|
|
|
return fts::fuzzy_match_simple(search_pattern, label_str ) ||
|
|
|
|
fts::fuzzy_match_simple(search_pattern, opt_key_str ) ;
|
|
|
|
}
|
|
|
|
|
2020-03-31 20:46:12 +00:00
|
|
|
bool SearchOptions::Option::fuzzy_match_simple(const wxString& search) const
|
2020-03-25 09:24:43 +00:00
|
|
|
{
|
|
|
|
char const* search_pattern = search.utf8_str();
|
2020-03-31 20:46:12 +00:00
|
|
|
return fuzzy_match_simple(search_pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SearchOptions::Option::fuzzy_match_simple(const std::string& search) const
|
|
|
|
{
|
|
|
|
char const* search_pattern = search.c_str();
|
|
|
|
return fuzzy_match_simple(search_pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SearchOptions::Option::fuzzy_match(char const* search_pattern, int& outScore)
|
|
|
|
{
|
2020-03-25 09:24:43 +00:00
|
|
|
char const* opt_key_str = opt_key.c_str();
|
|
|
|
char const* label_str = label.utf8_str();
|
|
|
|
|
|
|
|
return (fts::fuzzy_match(search_pattern, label_str , outScore) ||
|
|
|
|
fts::fuzzy_match(search_pattern, opt_key_str , outScore) );
|
2020-03-13 21:37:18 +00:00
|
|
|
}
|
|
|
|
|
2020-03-31 20:46:12 +00:00
|
|
|
bool SearchOptions::Option::fuzzy_match(const wxString& search, int& outScore)
|
|
|
|
{
|
|
|
|
char const* search_pattern = search.utf8_str();
|
|
|
|
return fuzzy_match(search_pattern, outScore);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SearchOptions::Option::fuzzy_match(const std::string& search, int& outScore)
|
|
|
|
{
|
|
|
|
char const* search_pattern = search.c_str();
|
|
|
|
return fuzzy_match(search_pattern, outScore);
|
|
|
|
}
|
|
|
|
|
2020-03-28 18:39:24 +00:00
|
|
|
void SearchOptions::Filter::get_label(const char** out_text) const
|
|
|
|
{
|
|
|
|
*out_text = label.utf8_str();
|
|
|
|
}
|
2020-03-13 21:37:18 +00:00
|
|
|
|
2020-04-03 08:01:23 +00:00
|
|
|
void SearchOptions::Filter::get_marked_label(const char** out_text) const
|
|
|
|
{
|
|
|
|
*out_text = marked_label.utf8_str();
|
|
|
|
}
|
|
|
|
|
2020-03-13 21:37:18 +00:00
|
|
|
template<class T>
|
|
|
|
void change_opt_key(std::string& opt_key, DynamicPrintConfig* config)
|
|
|
|
{
|
|
|
|
T* opt_cur = static_cast<T*>(config->option(opt_key));
|
|
|
|
if (opt_cur->values.size() > 0)
|
|
|
|
opt_key += "#" + std::to_string(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchOptions::append_options(DynamicPrintConfig* config, Preset::Type type, ConfigOptionMode mode)
|
|
|
|
{
|
|
|
|
for (std::string opt_key : config->keys())
|
|
|
|
{
|
|
|
|
const ConfigOptionDef& opt = config->def()->options.at(opt_key);
|
|
|
|
if (opt.mode > mode)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (type == Preset::TYPE_SLA_MATERIAL || type == Preset::TYPE_PRINTER)
|
|
|
|
switch (config->option(opt_key)->type())
|
|
|
|
{
|
|
|
|
case coInts: change_opt_key<ConfigOptionInts >(opt_key, config); break;
|
|
|
|
case coBools: change_opt_key<ConfigOptionBools >(opt_key, config); break;
|
|
|
|
case coFloats: change_opt_key<ConfigOptionFloats >(opt_key, config); break;
|
|
|
|
case coStrings: change_opt_key<ConfigOptionStrings >(opt_key, config); break;
|
|
|
|
case coPercents:change_opt_key<ConfigOptionPercents >(opt_key, config); break;
|
|
|
|
case coPoints: change_opt_key<ConfigOptionPoints >(opt_key, config); break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxString label;
|
|
|
|
if (!opt.category.empty())
|
|
|
|
label += _(opt.category) + " : ";
|
|
|
|
label += _(opt.full_label.empty() ? opt.label : opt.full_label);
|
|
|
|
|
2020-03-28 18:39:24 +00:00
|
|
|
if (!label.IsEmpty())
|
|
|
|
options.emplace_back(Option{ label, opt_key, opt.category, type });
|
2020-03-13 21:37:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-31 20:46:12 +00:00
|
|
|
// Wrap a string with ColorMarkerStart and ColorMarkerEnd symbols
|
|
|
|
static wxString wrap_string(const wxString& str)
|
|
|
|
{
|
|
|
|
return wxString::Format("%c%s%c", ImGui::ColorMarkerStart, str, ImGui::ColorMarkerEnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark a string using ColorMarkerStart and ColorMarkerEnd symbols
|
|
|
|
static void mark_string(wxString& str, const wxString& search_str)
|
|
|
|
{
|
|
|
|
// Try to find whole search string
|
|
|
|
if (str.Replace(search_str, wrap_string(search_str), false) != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Try to find whole capitalized search string
|
|
|
|
wxString search_str_capitalized = search_str.Capitalize();
|
|
|
|
if (str.Replace(search_str_capitalized, wrap_string(search_str_capitalized), false) != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// if search string is just a one letter now, there is no reason to continue
|
|
|
|
if (search_str.Len()==1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Split a search string for two strings (string without last letter and last letter)
|
|
|
|
// and repeat a function with new search strings
|
|
|
|
mark_string(str, search_str.SubString(0, search_str.Len() - 2));
|
|
|
|
mark_string(str, search_str.Last());
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear marked string from a redundant use of ColorMarkers
|
|
|
|
static void clear_marked_string(wxString& str)
|
|
|
|
{
|
|
|
|
// Check if the string has a several ColorMarkerStart in a row and replace them to only one, if any
|
|
|
|
wxString delete_string = wxString::Format("%c%c", ImGui::ColorMarkerStart, ImGui::ColorMarkerStart);
|
2020-04-03 12:29:57 +00:00
|
|
|
str.Replace(delete_string, ImGui::ColorMarkerStart, true);
|
|
|
|
// If there were several ColorMarkerStart in a row, it means there should be a several ColorMarkerStop in a row,
|
|
|
|
// replace them to only one
|
|
|
|
delete_string = wxString::Format("%c%c", ImGui::ColorMarkerEnd, ImGui::ColorMarkerEnd);
|
|
|
|
str.Replace(delete_string, ImGui::ColorMarkerEnd, true);
|
2020-03-31 20:46:12 +00:00
|
|
|
|
|
|
|
// And we should to remove redundant ColorMarkers, if they are in "End, Start" sequence in a row
|
|
|
|
delete_string = wxString::Format("%c%c", ImGui::ColorMarkerEnd, ImGui::ColorMarkerStart);
|
|
|
|
str.Replace(delete_string, wxEmptyString, true);
|
|
|
|
}
|
|
|
|
|
2020-03-28 18:39:24 +00:00
|
|
|
void SearchOptions::apply_filters(const std::string& search)
|
2020-03-25 09:24:43 +00:00
|
|
|
{
|
|
|
|
clear_filters();
|
2020-03-28 18:39:24 +00:00
|
|
|
|
|
|
|
bool full_list = search.empty();
|
2020-03-31 20:46:12 +00:00
|
|
|
|
|
|
|
for (size_t i=0; i < options.size(); i++)
|
|
|
|
{
|
|
|
|
if (full_list) {
|
2020-04-03 08:01:23 +00:00
|
|
|
filters.emplace_back(Filter{ options[i].label, options[i].label, i, 0 });
|
2020-03-31 20:46:12 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int score = 0;
|
|
|
|
if (options[i].fuzzy_match_simple(search)/*fuzzy_match(search, score)*/)
|
|
|
|
{
|
2020-04-03 08:01:23 +00:00
|
|
|
wxString marked_label = options[i].label;
|
|
|
|
mark_string(marked_label, from_u8(search));
|
|
|
|
clear_marked_string(marked_label);
|
2020-03-31 20:46:12 +00:00
|
|
|
|
2020-04-03 08:01:23 +00:00
|
|
|
filters.emplace_back(Filter{ options[i].label, marked_label, i, score });
|
2020-03-31 20:46:12 +00:00
|
|
|
}
|
2020-03-25 09:24:43 +00:00
|
|
|
}
|
2020-03-28 18:39:24 +00:00
|
|
|
|
|
|
|
if (!full_list)
|
|
|
|
sort_filters();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchOptions::init(std::vector<SearchInput> input_values)
|
|
|
|
{
|
|
|
|
clear_options();
|
|
|
|
for (auto i : input_values)
|
|
|
|
append_options(i.config, i.type, i.mode);
|
|
|
|
sort_options();
|
|
|
|
|
|
|
|
apply_filters("");
|
|
|
|
}
|
|
|
|
|
|
|
|
const SearchOptions::Option& SearchOptions::get_option(size_t pos_in_filter) const
|
|
|
|
{
|
|
|
|
assert(pos_in_filter != size_t(-1) && filters[pos_in_filter].option_idx != size_t(-1));
|
|
|
|
return options[filters[pos_in_filter].option_idx];
|
2020-03-25 09:24:43 +00:00
|
|
|
}
|
2020-04-03 08:01:23 +00:00
|
|
|
/*
|
|
|
|
SearchComboBox::SearchComboBox(wxWindow *parent, SearchOptions& search_list) :
|
2020-03-21 18:45:55 +00:00
|
|
|
wxBitmapComboBox(parent, wxID_ANY, _(L("Type here to search")) + dots, wxDefaultPosition, wxSize(25 * wxGetApp().em_unit(), -1)),
|
2020-04-03 08:01:23 +00:00
|
|
|
em_unit(wxGetApp().em_unit()),
|
|
|
|
search_list(search_list)
|
2020-03-13 21:37:18 +00:00
|
|
|
{
|
|
|
|
SetFont(wxGetApp().normal_font());
|
2020-03-21 18:45:55 +00:00
|
|
|
default_search_line = search_line = _(L("Type here to search")) + dots;
|
2020-03-13 21:37:18 +00:00
|
|
|
bmp = ScalableBitmap(this, "search");
|
|
|
|
|
|
|
|
Bind(wxEVT_COMBOBOX, [this](wxCommandEvent &evt) {
|
|
|
|
auto selected_item = this->GetSelection();
|
2020-04-03 08:01:23 +00:00
|
|
|
if (selected_item < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
wxGetApp().sidebar().jump_to_option(selected_item);
|
|
|
|
|
|
|
|
return;
|
2020-03-13 21:37:18 +00:00
|
|
|
SearchOptions::Option* opt = reinterpret_cast<SearchOptions::Option*>(this->GetClientData(selected_item));
|
|
|
|
wxGetApp().get_tab(opt->type)->activate_option(opt->opt_key, opt->category);
|
|
|
|
|
|
|
|
evt.StopPropagation();
|
|
|
|
|
|
|
|
SuppressUpdate su(this);
|
|
|
|
this->SetValue(search_line);
|
|
|
|
});
|
|
|
|
|
|
|
|
Bind(wxEVT_TEXT, [this](wxCommandEvent &e) {
|
2020-04-03 08:01:23 +00:00
|
|
|
/* if (prevent_update)
|
2020-03-13 21:37:18 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (this->IsTextEmpty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-03 08:01:23 +00:00
|
|
|
* / if (search_line != this->GetValue()) {
|
|
|
|
std::string& search_str = wxGetApp().sidebar().get_search_line();
|
|
|
|
search_str = into_u8(this->GetValue());
|
|
|
|
wxGetApp().sidebar().apply_search_filter();
|
2020-03-13 21:37:18 +00:00
|
|
|
update_combobox();
|
|
|
|
search_line = this->GetValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
e.Skip();
|
2020-03-21 18:45:55 +00:00
|
|
|
});
|
2020-03-13 21:37:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SearchComboBox::~SearchComboBox()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchComboBox::msw_rescale()
|
|
|
|
{
|
|
|
|
em_unit = wxGetApp().em_unit();
|
|
|
|
|
|
|
|
wxSize size = wxSize(25 * em_unit, -1);
|
|
|
|
|
|
|
|
// Set rescaled min height to correct layout
|
|
|
|
this->SetMinSize(size);
|
|
|
|
// Set rescaled size
|
|
|
|
this->SetSize(size);
|
|
|
|
|
|
|
|
update_combobox();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchComboBox::init(DynamicPrintConfig* config, Preset::Type type, ConfigOptionMode mode)
|
|
|
|
{
|
2020-03-25 09:24:43 +00:00
|
|
|
search_list.clear_options();
|
2020-03-13 21:37:18 +00:00
|
|
|
search_list.append_options(config, type, mode);
|
2020-03-25 09:24:43 +00:00
|
|
|
search_list.sort_options();
|
2020-03-13 21:37:18 +00:00
|
|
|
|
|
|
|
update_combobox();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchComboBox::init(std::vector<SearchInput> input_values)
|
|
|
|
{
|
2020-03-25 09:24:43 +00:00
|
|
|
search_list.clear_options();
|
2020-03-13 21:37:18 +00:00
|
|
|
for (auto i : input_values)
|
|
|
|
search_list.append_options(i.config, i.type, i.mode);
|
2020-03-25 09:24:43 +00:00
|
|
|
search_list.sort_options();
|
2020-03-13 21:37:18 +00:00
|
|
|
|
|
|
|
update_combobox();
|
|
|
|
}
|
|
|
|
|
2020-03-28 18:39:24 +00:00
|
|
|
void SearchComboBox::init(const SearchOptions& new_search_list)
|
|
|
|
{
|
|
|
|
search_list = new_search_list;
|
|
|
|
|
|
|
|
update_combobox();
|
|
|
|
}
|
|
|
|
|
2020-03-13 21:37:18 +00:00
|
|
|
void SearchComboBox::update_combobox()
|
|
|
|
{
|
2020-04-03 08:01:23 +00:00
|
|
|
this->Clear();
|
|
|
|
for (const SearchOptions::Filter& item : search_list.filters)
|
|
|
|
append(item.label);
|
|
|
|
|
|
|
|
// SuppressUpdate su(this);
|
|
|
|
// this->SetValue(default_search_line);
|
|
|
|
|
|
|
|
return;
|
2020-03-13 21:37:18 +00:00
|
|
|
wxString search_str = this->GetValue();
|
|
|
|
if (search_str.IsEmpty() || search_str == default_search_line)
|
|
|
|
// add whole options list to the controll
|
|
|
|
append_all_items();
|
|
|
|
else
|
|
|
|
append_items(search_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchComboBox::append_all_items()
|
|
|
|
{
|
|
|
|
this->Clear();
|
|
|
|
for (const SearchOptions::Option& item : search_list.options)
|
|
|
|
if (!item.label.IsEmpty())
|
|
|
|
append(item.label, (void*)&item);
|
|
|
|
|
|
|
|
SuppressUpdate su(this);
|
|
|
|
this->SetValue(default_search_line);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchComboBox::append_items(const wxString& search)
|
|
|
|
{
|
|
|
|
this->Clear();
|
2020-03-25 09:24:43 +00:00
|
|
|
/*
|
|
|
|
search_list.apply_filters(search);
|
|
|
|
for (auto filter : search_list.filters) {
|
|
|
|
auto it = std::lower_bound(search_list.options.begin(), search_list.options.end(), SearchOptions::Option{filter.label});
|
|
|
|
if (it != search_list.options.end())
|
|
|
|
append(it->label, (void*)(&(*it)));
|
|
|
|
}
|
2020-04-03 08:01:23 +00:00
|
|
|
* /
|
2020-03-13 21:37:18 +00:00
|
|
|
|
|
|
|
for (const SearchOptions::Option& option : search_list.options)
|
2020-03-31 20:46:12 +00:00
|
|
|
if (option.fuzzy_match_simple(search))
|
2020-03-13 21:37:18 +00:00
|
|
|
append(option.label, (void*)&option);
|
|
|
|
|
|
|
|
SuppressUpdate su(this);
|
|
|
|
this->SetValue(search);
|
|
|
|
this->SetInsertionPointEnd();
|
|
|
|
}
|
2020-04-03 08:01:23 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
//------------------------------------------
|
|
|
|
// PopupSearchList
|
|
|
|
//------------------------------------------
|
|
|
|
|
|
|
|
PopupSearchList::PopupSearchList(wxWindow* parent) :
|
2020-04-04 17:25:14 +00:00
|
|
|
wxPopupTransientWindow(parent, wxSTAY_ON_TOP| wxBORDER_NONE)
|
2020-04-03 08:01:23 +00:00
|
|
|
{
|
|
|
|
panel = new wxPanel(this, wxID_ANY);
|
|
|
|
|
|
|
|
int em_unit = wxGetApp().em_unit();
|
|
|
|
|
|
|
|
search_ctrl = new wxListCtrl(panel, wxID_ANY, wxDefaultPosition, wxSize(25 * em_unit, 35 * em_unit), wxLC_NO_HEADER | wxLC_REPORT);
|
|
|
|
search_ctrl->AppendColumn("");
|
|
|
|
search_ctrl->SetColumnWidth(0, 23 * em_unit);
|
|
|
|
search_ctrl->Bind(wxEVT_LIST_ITEM_SELECTED, &PopupSearchList::OnSelect, this);
|
|
|
|
|
|
|
|
wxBoxSizer* topSizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
|
2020-04-04 17:25:14 +00:00
|
|
|
wxTextCtrl *text = new wxTextCtrl(panel, wxID_ANY, "Brrrr");
|
|
|
|
text->Bind(wxEVT_ACTIVATE, [](wxEvent& e) {
|
|
|
|
int i=0; });
|
|
|
|
text->Bind(wxEVT_MOUSE_CAPTURE_CHANGED, [](wxEvent& e) {
|
|
|
|
int i = 0; });
|
|
|
|
text->Bind(wxEVT_LEFT_DOWN, [text](wxEvent& e) {
|
|
|
|
text->SetValue("mrrrrrty"); });
|
|
|
|
text->Bind(wxEVT_TEXT, [text](wxCommandEvent& e) {
|
|
|
|
text->SetSelection(2, 3); });
|
|
|
|
text->Bind(wxEVT_CHAR, [text](wxKeyEvent& e) {
|
|
|
|
text->SetFocus(); });
|
|
|
|
|
|
|
|
topSizer->Add(text, 0, wxEXPAND | wxALL, 2);
|
2020-04-03 08:01:23 +00:00
|
|
|
topSizer->Add(search_ctrl, 0, wxEXPAND | wxALL, 2);
|
|
|
|
|
|
|
|
panel->SetSizer(topSizer);
|
|
|
|
|
|
|
|
topSizer->Fit(panel);
|
|
|
|
SetClientSize(panel->GetSize());
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::Popup(wxWindow* WXUNUSED(focus))
|
|
|
|
{
|
|
|
|
wxPopupTransientWindow::Popup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::OnDismiss()
|
|
|
|
{
|
|
|
|
wxPopupTransientWindow::OnDismiss();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PopupSearchList::ProcessLeftDown(wxMouseEvent& event)
|
|
|
|
{
|
|
|
|
return wxPopupTransientWindow::ProcessLeftDown(event);
|
|
|
|
}
|
|
|
|
bool PopupSearchList::Show(bool show)
|
|
|
|
{
|
|
|
|
return wxPopupTransientWindow::Show(show);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::OnSize(wxSizeEvent& event)
|
|
|
|
{
|
|
|
|
event.Skip();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::OnSetFocus(wxFocusEvent& event)
|
|
|
|
{
|
|
|
|
event.Skip();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::OnKillFocus(wxFocusEvent& event)
|
|
|
|
{
|
|
|
|
event.Skip();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::OnSelect(wxListEvent& event)
|
|
|
|
{
|
|
|
|
int selection = event.GetIndex();
|
|
|
|
if (selection>=0)
|
|
|
|
wxGetApp().sidebar().jump_to_option(selection);
|
|
|
|
|
|
|
|
OnDismiss();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PopupSearchList::update_list(std::vector<SearchOptions::Filter>& filters)
|
|
|
|
{
|
|
|
|
search_ctrl->DeleteAllItems();
|
|
|
|
for (const SearchOptions::Filter& item : filters)
|
|
|
|
search_ctrl->InsertItem(search_ctrl->GetItemCount(), item.label);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//------------------------------------------
|
|
|
|
// SearchCtrl
|
|
|
|
//------------------------------------------
|
|
|
|
|
2020-04-03 10:51:20 +00:00
|
|
|
SearchCtrl::SearchCtrl(wxWindow* parent)
|
2020-04-03 08:01:23 +00:00
|
|
|
{
|
2020-04-04 17:25:14 +00:00
|
|
|
// popup_win = new PopupSearchList(parent);
|
2020-04-03 08:01:23 +00:00
|
|
|
box_sizer = new wxBoxSizer(wxHORIZONTAL);
|
2020-04-04 17:25:14 +00:00
|
|
|
/*
|
2020-04-03 08:01:23 +00:00
|
|
|
search_line = new wxTextCtrl(parent, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(25 * wxGetApp().em_unit(), -1), wxTE_PROCESS_ENTER);
|
|
|
|
search_line->Bind(wxEVT_TEXT, &SearchCtrl::OnInputText, this);
|
|
|
|
search_line->Bind(wxEVT_TEXT_ENTER, &SearchCtrl::PopupList, this);
|
|
|
|
|
|
|
|
search_btn = new ScalableButton(parent, wxID_ANY, "search");
|
|
|
|
search_btn->Bind(wxEVT_BUTTON, &SearchCtrl::PopupList, this);
|
|
|
|
|
|
|
|
box_sizer->Add(search_line, 0, wxALIGN_CENTER_VERTICAL);
|
|
|
|
box_sizer->AddSpacer(5);
|
|
|
|
box_sizer->Add(search_btn, 0, wxALIGN_CENTER_VERTICAL);
|
2020-04-04 17:25:14 +00:00
|
|
|
|
|
|
|
*/
|
|
|
|
default_string = _L("Type here to search");
|
|
|
|
|
|
|
|
comboCtrl = new wxComboCtrl(parent, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(25 * wxGetApp().em_unit(), -1), wxTE_PROCESS_ENTER);
|
|
|
|
comboCtrl->UseAltPopupWindow();
|
|
|
|
|
|
|
|
wxBitmap bmp_norm = create_scaled_bitmap("search_gray");
|
|
|
|
wxBitmap bmp_hov = create_scaled_bitmap("search");
|
|
|
|
comboCtrl->SetButtonBitmaps(bmp_norm, true, bmp_hov, bmp_hov, bmp_norm);
|
2020-04-05 20:11:45 +00:00
|
|
|
|
|
|
|
// popupListBox = new SearchComboPopup();
|
|
|
|
popupListCtrl = new SearchComboPopup_();
|
|
|
|
|
|
|
|
// It is important to call SetPopupControl() as soon as possible
|
|
|
|
// comboCtrl->SetPopupControl(popupListBox);
|
|
|
|
comboCtrl->SetPopupControl(popupListCtrl);
|
|
|
|
|
2020-04-04 17:25:14 +00:00
|
|
|
box_sizer->Add(comboCtrl, 0, wxALIGN_CENTER_VERTICAL);
|
|
|
|
|
2020-04-05 20:11:45 +00:00
|
|
|
// popupListBox->Bind(wxEVT_LISTBOX, &SearchCtrl::OnSelect, this);
|
|
|
|
// popupListCtrl->Bind(wxEVT_LIST_ITEM_SELECTED, &SearchCtrl::OnSelectCtrl, this);
|
|
|
|
popupListCtrl->Bind(wxEVT_LIST_ITEM_SELECTED, [](wxListEvent& event)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
});
|
2020-04-04 17:25:14 +00:00
|
|
|
|
|
|
|
comboCtrl->Bind(wxEVT_TEXT, &SearchCtrl::OnInputText, this);
|
|
|
|
comboCtrl->Bind(wxEVT_TEXT_ENTER, &SearchCtrl::PopupList, this);
|
|
|
|
comboCtrl->Bind(wxEVT_COMBOBOX_DROPDOWN, &SearchCtrl::PopupList, this);
|
|
|
|
|
|
|
|
comboCtrl->GetTextCtrl()->Bind(wxEVT_LEFT_DOWN, &SearchCtrl::OnLeftDown, this);
|
2020-04-03 08:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SearchCtrl::~SearchCtrl()
|
|
|
|
{
|
2020-04-03 10:51:20 +00:00
|
|
|
if (search_line)
|
|
|
|
search_line->Destroy();
|
2020-04-03 08:01:23 +00:00
|
|
|
if (search_btn)
|
|
|
|
search_btn->Destroy();
|
|
|
|
if (popup_win)
|
|
|
|
popup_win->Destroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchCtrl::OnInputText(wxCommandEvent& )
|
|
|
|
{
|
|
|
|
if (prevent_update)
|
|
|
|
return;
|
2020-04-04 17:25:14 +00:00
|
|
|
|
2020-04-03 08:01:23 +00:00
|
|
|
std::string& search_str = wxGetApp().sidebar().get_search_line();
|
2020-04-04 17:25:14 +00:00
|
|
|
// search_str = into_u8(search_line->GetValue());
|
|
|
|
search_str = into_u8(comboCtrl->GetValue());
|
2020-04-03 08:01:23 +00:00
|
|
|
wxGetApp().sidebar().apply_search_filter();
|
|
|
|
|
2020-04-04 17:25:14 +00:00
|
|
|
// popup_win->update_list(wxGetApp().sidebar().get_search_list().filters);
|
|
|
|
|
|
|
|
// update_list(wxGetApp().sidebar().get_search_list().filters);
|
2020-04-03 08:01:23 +00:00
|
|
|
}
|
|
|
|
|
2020-04-04 17:25:14 +00:00
|
|
|
void SearchCtrl::PopupList(wxCommandEvent& e)
|
2020-04-03 08:01:23 +00:00
|
|
|
{
|
2020-04-04 17:25:14 +00:00
|
|
|
/* popup_win->update_list(wxGetApp().sidebar().get_search_list().filters);
|
2020-04-03 08:01:23 +00:00
|
|
|
|
|
|
|
wxPoint pos = search_line->ClientToScreen(wxPoint(0, 0));
|
|
|
|
wxSize sz = search_line->GetSize();
|
|
|
|
pos.x -= sz.GetWidth();
|
|
|
|
popup_win->Position(pos, sz);
|
|
|
|
|
|
|
|
popup_win->Popup();
|
2020-04-04 17:25:14 +00:00
|
|
|
*/
|
|
|
|
update_list(wxGetApp().sidebar().get_search_list().filters);
|
|
|
|
e.Skip();
|
2020-04-03 08:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SearchCtrl::set_search_line(const std::string& line)
|
|
|
|
{
|
|
|
|
prevent_update = true;
|
2020-04-04 17:25:14 +00:00
|
|
|
// search_line->SetValue(line.empty() ? _L("Type here to search") : from_u8(line));
|
|
|
|
comboCtrl->SetText(line.empty() ? default_string : from_u8(line));
|
2020-04-03 08:01:23 +00:00
|
|
|
prevent_update = false;
|
2020-04-04 17:25:14 +00:00
|
|
|
|
2020-04-03 08:01:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SearchCtrl::msw_rescale()
|
|
|
|
{
|
|
|
|
wxSize size = wxSize(25 * wxGetApp().em_unit(), -1);
|
|
|
|
// Set rescaled min height to correct layout
|
|
|
|
search_line->SetMinSize(size);
|
|
|
|
// Set rescaled size
|
|
|
|
search_btn->msw_rescale();
|
2020-04-04 17:25:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
comboCtrl->SetButtonBitmaps(create_scaled_bitmap("search"));
|
2020-04-03 08:01:23 +00:00
|
|
|
}
|
|
|
|
|
2020-03-13 21:37:18 +00:00
|
|
|
|
2020-04-04 17:25:14 +00:00
|
|
|
void SearchCtrl::OnSelect(wxCommandEvent& event)
|
|
|
|
{
|
|
|
|
prevent_update = true;
|
|
|
|
|
|
|
|
int selection = event.GetSelection();
|
|
|
|
if (selection >= 0)
|
|
|
|
wxGetApp().sidebar().jump_to_option(selection);
|
|
|
|
|
|
|
|
prevent_update = false;
|
|
|
|
|
|
|
|
comboCtrl->Dismiss();
|
|
|
|
}
|
|
|
|
|
2020-04-05 20:11:45 +00:00
|
|
|
void SearchCtrl::OnSelectCtrl(wxListEvent& event)
|
|
|
|
{
|
|
|
|
prevent_update = true;
|
|
|
|
|
|
|
|
int selection = event.GetIndex();
|
|
|
|
if (selection >= 0)
|
|
|
|
wxGetApp().sidebar().jump_to_option(selection);
|
|
|
|
|
|
|
|
prevent_update = false;
|
|
|
|
|
|
|
|
comboCtrl->Dismiss();
|
|
|
|
}
|
|
|
|
|
2020-04-04 17:25:14 +00:00
|
|
|
void SearchCtrl::update_list(std::vector<SearchOptions::Filter>& filters)
|
|
|
|
{
|
2020-04-05 20:11:45 +00:00
|
|
|
/* popupListBox->Clear();
|
|
|
|
for (const SearchOptions::Filter& item : filters)
|
|
|
|
popupListBox->Append(item.label);*/
|
|
|
|
popupListCtrl->DeleteAllItems();
|
2020-04-04 17:25:14 +00:00
|
|
|
for (const SearchOptions::Filter& item : filters)
|
2020-04-05 20:11:45 +00:00
|
|
|
popupListCtrl->InsertItem(popupListCtrl->GetItemCount(), item.label);
|
2020-04-04 17:25:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SearchCtrl::OnLeftDown(wxEvent &event)
|
|
|
|
{
|
|
|
|
const wxString& str = comboCtrl->GetValue();
|
|
|
|
if (!str.IsEmpty() && str == default_string) {
|
|
|
|
prevent_update = true;
|
|
|
|
comboCtrl->SetValue("");
|
|
|
|
prevent_update = false;
|
|
|
|
}
|
|
|
|
event.Skip();
|
|
|
|
}
|
|
|
|
|
2020-03-13 21:37:18 +00:00
|
|
|
}} // namespace Slic3r::GUI
|