PrusaSlicer-NonPlainar/src/slic3r/GUI/wxExtensions.hpp

810 lines
26 KiB
C++
Raw Normal View History

2018-02-02 11:38:35 +00:00
#ifndef slic3r_GUI_wxExtensions_hpp_
#define slic3r_GUI_wxExtensions_hpp_
#include <wx/checklst.h>
#include <wx/combo.h>
#include <wx/dataview.h>
2018-05-04 16:32:20 +00:00
#include <wx/dc.h>
#include <wx/collpane.h>
#include <wx/wupdlock.h>
#include <wx/button.h>
2018-08-21 00:03:10 +00:00
#include <wx/slider.h>
2018-05-04 16:32:20 +00:00
#include <vector>
#include <set>
2018-10-22 13:18:05 +00:00
#include <functional>
2018-02-02 11:38:35 +00:00
wxMenuItem* append_menu_item(wxMenu* menu, int id, const wxString& string, const wxString& description,
std::function<void(wxCommandEvent& event)> cb, const std::string& icon = "", wxEvtHandler* event_handler = nullptr);
wxMenuItem* append_submenu(wxMenu* menu, wxMenu* sub_menu, int id, const wxString& string, const wxString& description, const std::string& icon = "");
2018-02-02 11:38:35 +00:00
class wxCheckListBoxComboPopup : public wxCheckListBox, public wxComboPopup
{
static const unsigned int DefaultWidth;
static const unsigned int DefaultHeight;
static const unsigned int DefaultItemHeight;
2018-02-02 11:38:35 +00:00
wxString m_text;
// Events sent on mouseclick are quite complex. Function OnListBoxSelection is supposed to pass the event to the checkbox, which works fine on
// Win. On OSX and Linux the events are generated differently - clicking on the checkbox square generates the event twice (and the square
// therefore seems not to respond).
// This enum is meant to save current state of affairs, i.e., if the event forwarding is ok to do or not. It is only used on Linux
// and OSX by some #ifdefs. It also stores information whether OnListBoxSelection is supposed to change the checkbox status,
// or if it changed status on its own already (which happens when the square is clicked). More comments in OnCheckListBox(...)
// There indeed is a better solution, maybe making a custom event used for the event passing to distinguish the original and passed message
// and blocking one of them on OSX and Linux. Feel free to refactor, but carefully test on all platforms.
enum class OnCheckListBoxFunction{
FreeToProceed,
RefuseToProceed,
WasRefusedLastTime
} m_check_box_events_status = OnCheckListBoxFunction::FreeToProceed;
2018-02-02 11:38:35 +00:00
public:
virtual bool Create(wxWindow* parent);
virtual wxWindow* GetControl();
virtual void SetStringValue(const wxString& value);
virtual wxString GetStringValue() const;
virtual wxSize GetAdjustedSize(int minWidth, int prefHeight, int maxHeight);
virtual void OnKeyEvent(wxKeyEvent& evt);
2018-02-02 11:38:35 +00:00
void OnCheckListBox(wxCommandEvent& evt);
void OnListBoxSelection(wxCommandEvent& evt);
};
// *** wxDataViewTreeCtrlComboBox ***
class wxDataViewTreeCtrlComboPopup: public wxDataViewTreeCtrl, public wxComboPopup
{
static const unsigned int DefaultWidth;
static const unsigned int DefaultHeight;
static const unsigned int DefaultItemHeight;
wxString m_text;
int m_cnt_open_items{0};
public:
virtual bool Create(wxWindow* parent);
virtual wxWindow* GetControl() { return this; }
virtual void SetStringValue(const wxString& value) { m_text = value; }
virtual wxString GetStringValue() const { return m_text; }
// virtual wxSize GetAdjustedSize(int minWidth, int prefHeight, int maxHeight);
virtual void OnKeyEvent(wxKeyEvent& evt);
void OnDataViewTreeCtrlSelection(wxCommandEvent& evt);
void SetItemsCnt(int cnt) { m_cnt_open_items = cnt; }
};
// *** PrusaCollapsiblePane ***
// ----------------------------------------------------------------------------
class PrusaCollapsiblePane : public wxCollapsiblePane
{
public:
PrusaCollapsiblePane() {}
PrusaCollapsiblePane(wxWindow *parent,
wxWindowID winid,
const wxString& label,
const wxPoint& pos = wxDefaultPosition,
const wxSize& size = wxDefaultSize,
long style = wxCP_DEFAULT_STYLE,
const wxValidator& val = wxDefaultValidator,
const wxString& name = wxCollapsiblePaneNameStr)
{
Create(parent, winid, label, pos, size, style, val, name);
}
~PrusaCollapsiblePane() {}
void OnStateChange(const wxSize& sz); //override/hide of OnStateChange from wxCollapsiblePane
virtual bool Show(bool show = true) override {
wxCollapsiblePane::Show(show);
OnStateChange(GetBestSize());
return true;
}
};
// *** PrusaCollapsiblePaneMSW *** used only #ifdef __WXMSW__
// ----------------------------------------------------------------------------
#ifdef __WXMSW__
class PrusaCollapsiblePaneMSW : public PrusaCollapsiblePane//wxCollapsiblePane
{
wxButton* m_pDisclosureTriangleButton = nullptr;
wxBitmap m_bmp_close;
wxBitmap m_bmp_open;
wxString m_strLabel;
public:
PrusaCollapsiblePaneMSW() {}
PrusaCollapsiblePaneMSW( wxWindow *parent,
wxWindowID winid,
const wxString& label,
const wxPoint& pos = wxDefaultPosition,
const wxSize& size = wxDefaultSize,
long style = wxCP_DEFAULT_STYLE,
const wxValidator& val = wxDefaultValidator,
const wxString& name = wxCollapsiblePaneNameStr)
{
Create(parent, winid, label, pos, size, style, val, name);
}
~PrusaCollapsiblePaneMSW() {}
2018-05-11 09:25:28 +00:00
bool Create(wxWindow *parent,
wxWindowID id,
const wxString& label,
const wxPoint& pos,
const wxSize& size,
long style,
const wxValidator& val,
const wxString& name);
void UpdateBtnBmp();
void SetLabel(const wxString &label) override;
bool Layout() override;
void Collapse(bool collapse) override;
};
#endif //__WXMSW__
2018-05-04 16:32:20 +00:00
// *****************************************************************************
2018-09-17 13:12:13 +00:00
// ----------------------------------------------------------------------------
// PrusaDataViewBitmapText: helper class used by PrusaBitmapTextRenderer
// ----------------------------------------------------------------------------
class PrusaDataViewBitmapText : public wxObject
{
public:
PrusaDataViewBitmapText(const wxString &text = wxEmptyString,
const wxBitmap& bmp = wxNullBitmap) :
m_text(text), m_bmp(bmp)
{ }
PrusaDataViewBitmapText(const PrusaDataViewBitmapText &other)
: wxObject(),
m_text(other.m_text),
m_bmp(other.m_bmp)
{ }
void SetText(const wxString &text) { m_text = text; }
wxString GetText() const { return m_text; }
2018-10-18 09:28:31 +00:00
void SetBitmap(const wxBitmap &bmp) { m_bmp = bmp; }
2018-09-17 13:12:13 +00:00
const wxBitmap &GetBitmap() const { return m_bmp; }
bool IsSameAs(const PrusaDataViewBitmapText& other) const {
return m_text == other.m_text && m_bmp.IsSameAs(other.m_bmp);
}
bool operator==(const PrusaDataViewBitmapText& other) const {
return IsSameAs(other);
}
bool operator!=(const PrusaDataViewBitmapText& other) const {
return !IsSameAs(other);
}
private:
wxString m_text;
wxBitmap m_bmp;
wxDECLARE_DYNAMIC_CLASS(PrusaDataViewBitmapText);
2018-09-17 13:12:13 +00:00
};
DECLARE_VARIANT_OBJECT(PrusaDataViewBitmapText)
2018-05-04 16:32:20 +00:00
// ----------------------------------------------------------------------------
// PrusaObjectDataViewModelNode: a node inside PrusaObjectDataViewModel
2018-05-04 16:32:20 +00:00
// ----------------------------------------------------------------------------
enum ItemType{
2018-10-18 08:40:26 +00:00
itUndef = 0,
itObject = 1,
itVolume = 2,
itInstanceRoot = 4,
itInstance = 8,
itSettings = 16
};
class PrusaObjectDataViewModelNode;
WX_DEFINE_ARRAY_PTR(PrusaObjectDataViewModelNode*, MyObjectTreeModelNodePtrArray);
2018-05-04 16:32:20 +00:00
class PrusaObjectDataViewModelNode
2018-05-04 16:32:20 +00:00
{
2018-06-18 12:20:29 +00:00
PrusaObjectDataViewModelNode* m_parent;
2018-05-04 16:32:20 +00:00
MyObjectTreeModelNodePtrArray m_children;
2018-09-17 13:12:13 +00:00
wxBitmap m_empty_bmp;
2018-10-18 08:40:26 +00:00
size_t m_volumes_cnt = 0;
2018-09-17 13:12:13 +00:00
std::vector< std::string > m_opt_categories;
2018-05-04 16:32:20 +00:00
public:
2018-10-18 09:28:31 +00:00
PrusaObjectDataViewModelNode(const wxString &name) {
2018-05-04 16:32:20 +00:00
m_parent = NULL;
m_name = name;
m_type = itObject;
#ifdef __WXGTK__
// it's necessary on GTK because of control have to know if this item will be container
// in another case you couldn't to add subitem for this item
// it will be produce "segmentation fault"
m_container = true;
#endif //__WXGTK__
set_object_action_icon();
2018-05-04 16:32:20 +00:00
}
PrusaObjectDataViewModelNode( PrusaObjectDataViewModelNode* parent,
const wxString& sub_obj_name,
2018-09-17 13:12:13 +00:00
const wxBitmap& bmp,
const wxString& extruder,
const int idx = -1 ) {
2018-05-04 16:32:20 +00:00
m_parent = parent;
m_name = sub_obj_name;
2018-09-17 13:12:13 +00:00
m_bmp = bmp;
m_type = itVolume;
m_idx = idx;
2018-09-17 13:12:13 +00:00
m_extruder = extruder;
#ifdef __WXGTK__
// it's necessary on GTK because of control have to know if this item will be container
// in another case you couldn't to add subitem for this item
// it will be produce "segmentation fault"
m_container = true;
#endif //__WXGTK__
set_part_action_icon();
2018-09-17 13:12:13 +00:00
}
2018-10-18 08:40:26 +00:00
PrusaObjectDataViewModelNode( PrusaObjectDataViewModelNode* parent, const ItemType type) :
2018-09-17 13:12:13 +00:00
m_parent(parent),
m_type(type),
m_extruder(wxEmptyString)
{
if (type == itSettings) {
m_name = "Settings to modified";
}
else if (type == itInstanceRoot) {
m_name = "Instances";
#ifdef __WXGTK__
m_container = true;
#endif //__WXGTK__
}
else if (type == itInstance) {
m_idx = parent->GetChildCount();
2018-10-18 08:40:26 +00:00
m_name = wxString::Format("Instance_%d", m_idx+1);
}
}
~PrusaObjectDataViewModelNode()
2018-05-04 16:32:20 +00:00
{
// free all our children nodes
size_t count = m_children.GetCount();
for (size_t i = 0; i < count; i++)
{
PrusaObjectDataViewModelNode *child = m_children[i];
2018-05-04 16:32:20 +00:00
delete child;
}
}
wxString m_name;
2018-09-17 13:12:13 +00:00
wxBitmap& m_bmp = m_empty_bmp;
ItemType m_type;
int m_idx = -1;
2018-05-04 16:32:20 +00:00
bool m_container = false;
wxString m_extruder = "default";
wxBitmap m_action_icon;
2018-05-04 16:32:20 +00:00
bool IsContainer() const
{
return m_container;
}
PrusaObjectDataViewModelNode* GetParent()
2018-05-04 16:32:20 +00:00
{
return m_parent;
}
MyObjectTreeModelNodePtrArray& GetChildren()
{
return m_children;
}
PrusaObjectDataViewModelNode* GetNthChild(unsigned int n)
2018-05-04 16:32:20 +00:00
{
return m_children.Item(n);
}
void Insert(PrusaObjectDataViewModelNode* child, unsigned int n)
2018-05-04 16:32:20 +00:00
{
2018-09-17 13:12:13 +00:00
if (!m_container)
m_container = true;
2018-05-04 16:32:20 +00:00
m_children.Insert(child, n);
}
void Append(PrusaObjectDataViewModelNode* child)
2018-05-04 16:32:20 +00:00
{
if (!m_container)
m_container = true;
m_children.Add(child);
}
void RemoveAllChildren()
{
if (GetChildCount() == 0)
return;
for (size_t id = GetChildCount() - 1; id >= 0; --id)
{
if (m_children.Item(id)->GetChildCount() > 0)
m_children[id]->RemoveAllChildren();
auto node = m_children[id];
m_children.RemoveAt(id);
delete node;
}
}
2018-05-29 22:36:44 +00:00
size_t GetChildCount() const
2018-05-04 16:32:20 +00:00
{
return m_children.GetCount();
}
bool SetValue(const wxVariant &variant, unsigned int col)
{
switch (col)
{
case 0:{
2018-09-17 13:12:13 +00:00
PrusaDataViewBitmapText data;
data << variant;
2018-09-17 13:12:13 +00:00
m_bmp = data.GetBitmap();
m_name = data.GetText();
return true;}
case 1:
m_extruder = variant.GetString();
return true;
2018-10-18 09:28:31 +00:00
case 2:
m_action_icon << variant;
return true;
default:
printf("MyObjectTreeModel::SetValue: wrong column");
}
return false;
}
2018-09-17 13:12:13 +00:00
void SetBitmap(const wxBitmap &icon)
{
m_bmp = icon;
}
2018-06-18 12:20:29 +00:00
ItemType GetType() const {
return m_type;
}
void SetIdx(const int& idx) {
m_idx = idx;
2018-10-18 08:40:26 +00:00
// update name if this node is instance
if (m_type == itInstance)
m_name = wxString::Format("Instance_%d", m_idx + 1);
}
2018-10-18 08:40:26 +00:00
int GetIdx() const {
return m_idx;
2018-06-18 12:20:29 +00:00
}
2018-06-19 10:24:16 +00:00
// use this function only for childrens
void AssignAllVal(PrusaObjectDataViewModelNode& from_node)
{
// ! Don't overwrite other values because of equality of this values for all children --
m_name = from_node.m_name;
m_bmp = from_node.m_bmp;
m_idx = from_node.m_idx;
m_extruder = from_node.m_extruder;
m_type = from_node.m_type;
2018-06-19 10:24:16 +00:00
}
bool SwapChildrens(int frst_id, int scnd_id) {
if (GetChildCount() < 2 ||
frst_id < 0 || frst_id >= GetChildCount() ||
scnd_id < 0 || scnd_id >= GetChildCount())
return false;
PrusaObjectDataViewModelNode new_scnd = *GetNthChild(frst_id);
PrusaObjectDataViewModelNode new_frst = *GetNthChild(scnd_id);
new_scnd.m_idx = m_children.Item(scnd_id)->m_idx;
new_frst.m_idx = m_children.Item(frst_id)->m_idx;
2018-06-19 10:24:16 +00:00
m_children.Item(frst_id)->AssignAllVal(new_frst);
m_children.Item(scnd_id)->AssignAllVal(new_scnd);
return true;
}
// Set action icons for node
void set_object_action_icon();
void set_part_action_icon();
2018-09-17 13:12:13 +00:00
bool update_settings_digest(const std::vector<std::string>& categories);
2018-10-18 08:40:26 +00:00
private:
friend class PrusaObjectDataViewModel;
2018-05-04 16:32:20 +00:00
};
// ----------------------------------------------------------------------------
// PrusaObjectDataViewModel
2018-05-04 16:32:20 +00:00
// ----------------------------------------------------------------------------
class PrusaObjectDataViewModel :public wxDataViewModel
2018-05-04 16:32:20 +00:00
{
std::vector<PrusaObjectDataViewModelNode*> m_objects;
std::vector<wxBitmap*> m_volume_bmps;
2018-05-04 16:32:20 +00:00
public:
2018-09-17 13:12:13 +00:00
PrusaObjectDataViewModel();
~PrusaObjectDataViewModel();
2018-05-04 16:32:20 +00:00
wxDataViewItem Add(const wxString &name);
2018-10-18 08:40:26 +00:00
wxDataViewItem AddVolumeChild(const wxDataViewItem &parent_item,
const wxString &name,
const int volume_type,
2018-09-17 13:12:13 +00:00
const int extruder = 0,
const bool create_frst_child = true);
2018-09-17 13:12:13 +00:00
wxDataViewItem AddSettingsChild(const wxDataViewItem &parent_item);
2018-10-18 08:40:26 +00:00
wxDataViewItem AddInstanceChild(const wxDataViewItem &parent_item, size_t num);
wxDataViewItem Delete(const wxDataViewItem &item);
2018-10-18 08:40:26 +00:00
wxDataViewItem DeleteLastInstance(const wxDataViewItem &parent_item, size_t num);
void DeleteAll();
2018-07-31 13:31:12 +00:00
void DeleteChildren(wxDataViewItem& parent);
void DeleteVolumeChildren(wxDataViewItem& parent);
wxDataViewItem GetItemById(int obj_idx);
2018-09-17 13:12:13 +00:00
wxDataViewItem GetItemByVolumeId(int obj_idx, int volume_idx);
wxDataViewItem GetItemByInstanceId(int obj_idx, int inst_idx);
2018-10-16 11:56:28 +00:00
int GetIdByItem(const wxDataViewItem& item);
2018-10-18 08:40:26 +00:00
int GetIdByItemAndType(const wxDataViewItem& item, const ItemType type) const;
int GetVolumeIdByItem(const wxDataViewItem& item) const;
int GetInstanceIdByItem(const wxDataViewItem& item) const;
void GetItemInfo(const wxDataViewItem& item, ItemType& type, int& obj_idx, int& idx);
bool IsEmpty() { return m_objects.empty(); }
2018-05-04 16:32:20 +00:00
// helper method for wxLog
wxString GetName(const wxDataViewItem &item) const;
2018-09-17 13:12:13 +00:00
wxBitmap& GetBitmap(const wxDataViewItem &item) const;
2018-05-04 16:32:20 +00:00
// helper methods to change the model
virtual unsigned int GetColumnCount() const override { return 3;}
virtual wxString GetColumnType(unsigned int col) const override{ return wxT("string"); }
virtual void GetValue(wxVariant &variant,
const wxDataViewItem &item, unsigned int col) const override;
virtual bool SetValue(const wxVariant &variant,
const wxDataViewItem &item, unsigned int col) override;
bool SetValue(const wxVariant &variant, const int item_idx, unsigned int col);
2018-05-04 16:32:20 +00:00
// wxDataViewItem MoveChildUp(const wxDataViewItem &item);
// wxDataViewItem MoveChildDown(const wxDataViewItem &item);
// For parent move child from cur_volume_id place to new_volume_id
// Remaining items will moved up/down accordingly
wxDataViewItem ReorganizeChildren(int cur_volume_id,
int new_volume_id,
const wxDataViewItem &parent);
2018-06-19 10:24:16 +00:00
2018-09-17 13:12:13 +00:00
virtual bool IsEnabled(const wxDataViewItem &item, unsigned int col) const override;
2018-05-04 16:32:20 +00:00
virtual wxDataViewItem GetParent(const wxDataViewItem &item) const override;
2018-10-18 08:40:26 +00:00
// get object item
wxDataViewItem GetTopParent(const wxDataViewItem &item) const;
2018-05-04 16:32:20 +00:00
virtual bool IsContainer(const wxDataViewItem &item) const override;
virtual unsigned int GetChildren(const wxDataViewItem &parent,
wxDataViewItemArray &array) const override;
// Is the container just a header or an item with all columns
// In our case it is an item with all columns
virtual bool HasContainerColumns(const wxDataViewItem& WXUNUSED(item)) const override { return true; }
2018-09-17 13:12:13 +00:00
ItemType GetItemType(const wxDataViewItem &item) const ;
wxDataViewItem GetItemByType(const wxDataViewItem &parent_item, ItemType type) const;
wxDataViewItem GetSettingsItem(const wxDataViewItem &item) const;
wxDataViewItem GetInstanceRootItem(const wxDataViewItem &item) const;
2018-09-17 13:12:13 +00:00
bool IsSettingsItem(const wxDataViewItem &item) const;
void UpdateSettingsDigest(const wxDataViewItem &item, const std::vector<std::string>& categories);
void SetVolumeBitmaps(const std::vector<wxBitmap*>& volume_bmps) { m_volume_bmps = volume_bmps; }
void SetVolumeType(const wxDataViewItem &item, const int type);
2018-05-04 16:32:20 +00:00
};
2018-09-17 13:12:13 +00:00
// ----------------------------------------------------------------------------
// PrusaBitmapTextRenderer
// ----------------------------------------------------------------------------
class PrusaBitmapTextRenderer : public wxDataViewCustomRenderer
{
public:
PrusaBitmapTextRenderer( wxDataViewCellMode mode = wxDATAVIEW_CELL_INERT,
int align = wxDVR_DEFAULT_ALIGNMENT):
wxDataViewCustomRenderer(wxT("PrusaDataViewBitmapText"), mode, align) {}
2018-09-17 13:12:13 +00:00
bool SetValue(const wxVariant &value);
bool GetValue(wxVariant &value) const;
virtual bool Render(wxRect cell, wxDC *dc, int state);
virtual wxSize GetSize() const;
virtual bool HasEditorCtrl() const { return false; }
private:
PrusaDataViewBitmapText m_value;
};
2018-05-04 16:32:20 +00:00
// ----------------------------------------------------------------------------
// MyCustomRenderer
// ----------------------------------------------------------------------------
class MyCustomRenderer : public wxDataViewCustomRenderer
{
public:
// This renderer can be either activatable or editable, for demonstration
// purposes. In real programs, you should select whether the user should be
// able to activate or edit the cell and it doesn't make sense to switch
// between the two -- but this is just an example, so it doesn't stop us.
explicit MyCustomRenderer(wxDataViewCellMode mode)
: wxDataViewCustomRenderer("string", mode, wxALIGN_CENTER)
{ }
virtual bool Render(wxRect rect, wxDC *dc, int state) override/*wxOVERRIDE*/
{
dc->SetBrush(*wxLIGHT_GREY_BRUSH);
dc->SetPen(*wxTRANSPARENT_PEN);
rect.Deflate(2);
dc->DrawRoundedRectangle(rect, 5);
RenderText(m_value,
0, // no offset
wxRect(dc->GetTextExtent(m_value)).CentreIn(rect),
dc,
state);
return true;
}
virtual bool ActivateCell(const wxRect& WXUNUSED(cell),
wxDataViewModel *WXUNUSED(model),
const wxDataViewItem &WXUNUSED(item),
unsigned int WXUNUSED(col),
const wxMouseEvent *mouseEvent) override/*wxOVERRIDE*/
{
wxString position;
if (mouseEvent)
position = wxString::Format("via mouse at %d, %d", mouseEvent->m_x, mouseEvent->m_y);
else
position = "from keyboard";
// wxLogMessage("MyCustomRenderer ActivateCell() %s", position);
return false;
}
virtual wxSize GetSize() const override/*wxOVERRIDE*/
{
return wxSize(60, 20);
}
virtual bool SetValue(const wxVariant &value) override/*wxOVERRIDE*/
{
m_value = value.GetString();
return true;
}
virtual bool GetValue(wxVariant &WXUNUSED(value)) const override/*wxOVERRIDE*/{ return true; }
virtual bool HasEditorCtrl() const override/*wxOVERRIDE*/{ return true; }
virtual wxWindow*
CreateEditorCtrl(wxWindow* parent,
wxRect labelRect,
const wxVariant& value) override/*wxOVERRIDE*/
{
wxTextCtrl* text = new wxTextCtrl(parent, wxID_ANY, value,
labelRect.GetPosition(),
labelRect.GetSize(),
wxTE_PROCESS_ENTER);
text->SetInsertionPointEnd();
return text;
}
virtual bool
GetValueFromEditorCtrl(wxWindow* ctrl, wxVariant& value) override/*wxOVERRIDE*/
{
wxTextCtrl* text = wxDynamicCast(ctrl, wxTextCtrl);
if (!text)
return false;
value = text->GetValue();
return true;
}
private:
wxString m_value;
};
2018-08-29 09:21:22 +00:00
// ----------------------------------------------------------------------------
// PrusaDoubleSlider
// ----------------------------------------------------------------------------
2018-08-21 00:03:10 +00:00
enum SelectedSlider {
ssUndef,
ssLower,
ssHigher
};
2018-08-23 11:01:18 +00:00
enum TicksAction{
taOnIcon,
2018-08-23 11:01:18 +00:00
taAdd,
taDel
};
2018-08-21 00:03:10 +00:00
class PrusaDoubleSlider : public wxControl
{
public:
PrusaDoubleSlider(
wxWindow *parent,
wxWindowID id,
int lowerValue,
int higherValue,
int minValue,
int maxValue,
const wxPoint& pos = wxDefaultPosition,
const wxSize& size = wxDefaultSize,
2018-09-17 13:12:13 +00:00
long style = wxSL_VERTICAL,
2018-08-21 00:03:10 +00:00
const wxValidator& val = wxDefaultValidator,
const wxString& name = wxEmptyString);
2018-10-31 11:56:08 +00:00
~PrusaDoubleSlider() {}
2018-08-21 00:03:10 +00:00
int GetLowerValue() const {
2018-08-21 00:03:10 +00:00
return m_lower_value;
}
int GetHigherValue() const {
2018-08-21 00:03:10 +00:00
return m_higher_value;
}
int GetActiveValue() const;
2018-09-17 13:12:13 +00:00
double GetLowerValueD() const { return get_double_value(ssLower); }
double GetHigherValueD() const { return get_double_value(ssHigher); }
2018-08-22 12:24:30 +00:00
wxSize DoGetBestSize() const override;
void SetLowerValue(const int lower_val);
void SetHigherValue(const int higher_val);
void SetMaxValue(const int max_value);
void SetKoefForLabels(const double koef) {
m_label_koef = koef;
}
void SetSliderValues(const std::vector<std::pair<int, double>>& values) {
m_values = values;
}
2018-09-17 13:12:13 +00:00
void ChangeOneLayerLock();
2018-08-21 00:03:10 +00:00
2018-10-31 11:56:08 +00:00
void OnPaint(wxPaintEvent& ) { render();}
2018-08-21 00:03:10 +00:00
void OnLeftDown(wxMouseEvent& event);
void OnMotion(wxMouseEvent& event);
void OnLeftUp(wxMouseEvent& event);
2018-10-31 11:56:08 +00:00
void OnEnterWin(wxMouseEvent& event) { enter_window(event, true); }
void OnLeaveWin(wxMouseEvent& event) { enter_window(event, false); }
2018-08-21 00:03:10 +00:00
void OnWheel(wxMouseEvent& event);
void OnKeyDown(wxKeyEvent &event);
void OnKeyUp(wxKeyEvent &event);
void OnRightDown(wxMouseEvent& event);
void OnRightUp(wxMouseEvent& event);
2018-08-21 00:03:10 +00:00
protected:
void render();
void draw_focus_rect();
void draw_action_icon(wxDC& dc, const wxPoint pt_beg, const wxPoint pt_end);
2018-08-22 12:24:30 +00:00
void draw_scroll_line(wxDC& dc, const int lower_pos, const int higher_pos);
void draw_thumb(wxDC& dc, const wxCoord& pos_coord, const SelectedSlider& selection);
void draw_thumbs(wxDC& dc, const wxCoord& lower_pos, const wxCoord& higher_pos);
void draw_ticks(wxDC& dc);
void draw_one_layer_icon(wxDC& dc);
2018-08-22 12:24:30 +00:00
void draw_thumb_item(wxDC& dc, const wxPoint& pos, const SelectedSlider& selection);
2018-08-23 11:01:18 +00:00
void draw_info_line_with_icon(wxDC& dc, const wxPoint& pos, SelectedSlider selection);
2018-08-22 12:24:30 +00:00
void draw_thumb_text(wxDC& dc, const wxPoint& pos, const SelectedSlider& selection) const;
void update_thumb_rect(const wxCoord& begin_x, const wxCoord& begin_y, const SelectedSlider& selection);
2018-08-21 00:03:10 +00:00
void detect_selected_slider(const wxPoint& pt, const bool is_mouse_wheel = false);
2018-08-22 12:24:30 +00:00
void correct_lower_value();
void correct_higher_value();
void move_current_thumb(const bool condition);
2018-08-23 11:01:18 +00:00
void action_tick(const TicksAction action);
void enter_window(wxMouseEvent& event, const bool enter);
2018-08-22 12:24:30 +00:00
2018-08-21 00:03:10 +00:00
bool is_point_in_rect(const wxPoint& pt, const wxRect& rect);
bool is_horizontal() const { return m_style == wxSL_HORIZONTAL; }
2018-08-22 12:24:30 +00:00
double get_scroll_step();
wxString get_label(const SelectedSlider& selection) const;
void get_lower_and_higher_position(int& lower_pos, int& higher_pos);
int get_value_from_position(const wxCoord x, const wxCoord y);
wxCoord get_position_from_value(const int value);
wxSize get_size();
void get_size(int *w, int *h);
2018-09-17 13:12:13 +00:00
double get_double_value(const SelectedSlider& selection) const;
2018-05-04 16:32:20 +00:00
2018-08-21 00:03:10 +00:00
private:
int m_min_value;
int m_max_value;
int m_lower_value;
int m_higher_value;
wxBitmap m_bmp_thumb_higher;
wxBitmap m_bmp_thumb_lower;
wxBitmap m_bmp_add_tick_on;
wxBitmap m_bmp_add_tick_off;
wxBitmap m_bmp_del_tick_on;
wxBitmap m_bmp_del_tick_off;
wxBitmap m_bmp_one_layer_lock_on;
wxBitmap m_bmp_one_layer_lock_off;
wxBitmap m_bmp_one_layer_unlock_on;
wxBitmap m_bmp_one_layer_unlock_off;
2018-08-21 00:03:10 +00:00
SelectedSlider m_selection;
bool m_is_left_down = false;
bool m_is_right_down = false;
bool m_is_one_layer = false;
bool m_is_focused = false;
2018-08-23 11:01:18 +00:00
bool m_is_action_icon_focesed = false;
bool m_is_one_layer_icon_focesed = false;
2018-08-21 00:03:10 +00:00
wxRect m_rect_lower_thumb;
wxRect m_rect_higher_thumb;
2018-08-23 11:01:18 +00:00
wxRect m_rect_tick_action;
wxRect m_rect_one_layer_icon;
2018-08-22 12:24:30 +00:00
wxSize m_thumb_size;
2018-08-23 11:01:18 +00:00
int m_tick_icon_dim;
int m_lock_icon_dim;
2018-08-21 00:03:10 +00:00
long m_style;
float m_label_koef = 1.0;
2018-08-21 00:03:10 +00:00
// control's view variables
wxCoord SLIDER_MARGIN; // margin around slider
wxPen DARK_ORANGE_PEN;
wxPen ORANGE_PEN;
wxPen LIGHT_ORANGE_PEN;
wxPen DARK_GREY_PEN;
wxPen GREY_PEN;
wxPen LIGHT_GREY_PEN;
std::vector<wxPen*> line_pens;
std::vector<wxPen*> segm_pens;
std::set<int> m_ticks;
std::vector<std::pair<int,double>> m_values;
2018-08-21 00:03:10 +00:00
};
// ----------------------------------------------------------------------------
// PrusaLockButton
// ----------------------------------------------------------------------------
class PrusaLockButton : public wxButton
{
public:
PrusaLockButton(
wxWindow *parent,
wxWindowID id,
const wxPoint& pos = wxDefaultPosition,
const wxSize& size = wxDefaultSize);
2018-10-31 11:56:08 +00:00
~PrusaLockButton() {}
void OnButton(wxCommandEvent& event);
2018-10-31 11:56:08 +00:00
void OnEnterBtn(wxMouseEvent& event) { enter_button(true); event.Skip(); }
void OnLeaveBtn(wxMouseEvent& event) { enter_button(false); event.Skip(); }
bool IsLocked() const { return m_is_pushed; }
protected:
void enter_button(const bool enter);
private:
bool m_is_pushed = false;
wxBitmap m_bmp_lock_on;
wxBitmap m_bmp_lock_off;
wxBitmap m_bmp_unlock_on;
wxBitmap m_bmp_unlock_off;
int m_lock_icon_dim;
};
2018-08-29 09:21:22 +00:00
// ******************************* EXPERIMENTS **********************************************
2018-07-30 10:17:32 +00:00
// ******************************************************************************************
2018-05-04 16:32:20 +00:00
2018-02-02 11:38:35 +00:00
#endif // slic3r_GUI_wxExtensions_hpp_