#include "SwitchButton.hpp" #include "Label.hpp" #include "StaticBox.hpp" #include "../wxExtensions.hpp" #include "../GUI_App.hpp" #include "slic3r/GUI/I18N.hpp" #include "../Utils/MacDarkMode.hpp" #include "../Utils/WxFontUtils.hpp" #ifdef __APPLE__ #include "libslic3r/MacUtils.hpp" #endif #include #include #include wxDEFINE_EVENT(wxCUSTOMEVT_SWITCH_POS, wxCommandEvent); SwitchButton::SwitchButton(wxWindow* parent, wxWindowID id) : wxBitmapToggleButton(parent, id, wxNullBitmap, wxDefaultPosition, wxDefaultSize, wxBORDER_NONE | wxBU_EXACTFIT) , m_on(this, "toggle_on", 16) , m_off(this, "toggle_off", 16) , text_color(std::pair{0xfffffe, (int) StateColor::Checked}, std::pair{0x6B6B6B, (int) StateColor::Normal}) , track_color(0xD9D9D9) , thumb_color(std::pair{0x009688, (int) StateColor::Checked}, std::pair{0xD9D9D9, (int) StateColor::Normal}) { SetBackgroundColour(StaticBox::GetParentBackgroundColor(parent)); Bind(wxEVT_TOGGLEBUTTON, [this](auto& e) { update(); e.Skip(); }); SetFont(Label::Body_12); Rescale(); } void SwitchButton::SetLabels(wxString const& lbl_on, wxString const& lbl_off) { labels[0] = lbl_on; labels[1] = lbl_off; Rescale(); } void SwitchButton::SetTextColor(StateColor const& color) { text_color = color; Rescale(); } void SwitchButton::SetTextColor2(StateColor const &color) { text_color2 = color; Rescale(); } void SwitchButton::SetTrackColor(StateColor const& color) { track_color = color; Rescale(); } void SwitchButton::SetThumbColor(StateColor const& color) { thumb_color = color; Rescale(); } void SwitchButton::SetValue(bool value) { if (value != GetValue()) { wxBitmapToggleButton::SetValue(value); update(); } } bool SwitchButton::SetBackgroundColour(const wxColour& colour) { if (wxBitmapToggleButton::SetBackgroundColour(colour)) { Rescale(); return true; } return false; } void SwitchButton::Rescale() { if (labels[0].IsEmpty()) { m_on.msw_rescale(); m_off.msw_rescale(); } else { wxBitmapToggleButton::SetBackgroundColour(StaticBox::GetParentBackgroundColor(GetParent())); #ifdef __WXOSX__ auto scale = Slic3r::GUI::mac_max_scaling_factor(); int BS = (int) scale; #else constexpr int BS = 1; #endif wxSize thumbSize; wxSize trackSize; wxClientDC dc(this); #ifdef __WXOSX__ dc.SetFont(dc.GetFont().Scaled(scale)); #endif wxSize textSize[2]; { textSize[0] = dc.GetTextExtent(labels[0]); textSize[1] = dc.GetTextExtent(labels[1]); } float fontScale = 0; { thumbSize = textSize[0]; auto size = textSize[1]; if (size.x > thumbSize.x) thumbSize.x = size.x; else size.x = thumbSize.x; thumbSize.x += BS * 12; thumbSize.y += BS * 6; trackSize.x = thumbSize.x + size.x + BS * 10; trackSize.y = thumbSize.y + BS * 2; auto maxWidth = GetMaxWidth(); #ifdef __WXOSX__ maxWidth *= scale; #endif if (trackSize.x > maxWidth) { fontScale = float(maxWidth) / trackSize.x; thumbSize.x -= (trackSize.x - maxWidth) / 2; trackSize.x = maxWidth; } } for (int i = 0; i < 2; ++i) { wxMemoryDC memdc(&dc); #ifdef __WXMSW__ wxBitmap bmp(trackSize.x, trackSize.y); memdc.SelectObject(bmp); memdc.SetBackground(wxBrush(GetBackgroundColour())); memdc.Clear(); #else wxImage image(trackSize); image.InitAlpha(); memset(image.GetAlpha(), 0, trackSize.GetWidth() * trackSize.GetHeight()); wxBitmap bmp(std::move(image)); memdc.SelectObject(bmp); #endif memdc.SetFont(dc.GetFont()); #ifdef __WXMSW__ const double scale = GetDPIScaleFactor(); fontScale = scale; #endif if (fontScale) { memdc.SetFont(dc.GetFont().Scaled(fontScale)); textSize[0] = memdc.GetTextExtent(labels[0]); textSize[1] = memdc.GetTextExtent(labels[1]); } auto state = i == 0 ? StateColor::Enabled : (StateColor::Checked | StateColor::Enabled); { #ifdef __WXMSW__ wxGCDC dc2(memdc); #else wxDC &dc2(memdc); #endif dc2.SetBrush(wxBrush(track_color.colorForStates(state))); dc2.SetPen(wxPen(track_color.colorForStates(state))); dc2.DrawRoundedRectangle(wxRect({0, 0}, trackSize), trackSize.y / 2.0); dc2.SetBrush(wxBrush(thumb_color.colorForStates(StateColor::Checked | StateColor::Enabled))); dc2.SetPen(wxPen(thumb_color.colorForStates(StateColor::Checked | StateColor::Enabled))); dc2.DrawRoundedRectangle(wxRect({ i == 0 ? BS : (trackSize.x - thumbSize.x - BS), BS}, thumbSize), thumbSize.y / 2.0); } memdc.SetTextForeground(text_color.colorForStates(state ^ StateColor::Checked)); auto text_y = BS + (thumbSize.y - textSize[0].y) / 2; #ifdef __APPLE__ if (Slic3r::is_mac_version_15()) { text_y -= FromDIP(2); } #endif memdc.DrawText(labels[0], {BS + (thumbSize.x - textSize[0].x) / 2, text_y}); memdc.SetTextForeground(text_color2.count() == 0 ? text_color.colorForStates(state) : text_color2.colorForStates(state)); auto text_y_1 = BS + (thumbSize.y - textSize[1].y) / 2; #ifdef __APPLE__ if (Slic3r::is_mac_version_15()) { text_y_1 -= FromDIP(2); } #endif memdc.DrawText(labels[1], {trackSize.x - thumbSize.x - BS + (thumbSize.x - textSize[1].x) / 2, text_y_1}); memdc.SelectObject(wxNullBitmap); #ifdef __WXOSX__ bmp = wxBitmap(bmp.ConvertToImage(), -1, scale); #elif defined(__WXMSW__) bmp.SetScaleFactor(scale); // ORCA #endif (i == 0 ? m_off : m_on).bmp() = bmp; } } update(); #ifdef __WXGTK__ wxSize bestSize = GetBestSize(); bestSize.IncTo(m_on.GetBmpSize()); SetSize(bestSize); SetMinSize(bestSize); #else SetSize(m_on.GetBmpSize()); #endif } void SwitchButton::update() { SetBitmap((GetValue() ? m_on : m_off).bmp()); } ModeSwitchButton::ModeSwitchButton(wxWindow* parent, wxWindowID id) { background_color = StateColor( std::make_pair(wxColour(0xF1, 0xF1, 0xF1), (int) StateColor::Disabled), std::make_pair(wxColour(0xE3, 0xE3, 0xE3), (int) StateColor::Pressed), std::make_pair(wxColour(0xD9, 0xD9, 0xD9), (int) StateColor::Normal)); border_color = StateColor( std::make_pair(wxColour(0xEA, 0xEA, 0xEA), (int) StateColor::Disabled), std::make_pair(wxColour(0xBC, 0xBC, 0xBC), (int) StateColor::Hovered), std::make_pair(wxColour(0xC8, 0xC8, 0xC8), (int) StateColor::Focused), std::make_pair(wxColour(0xCE, 0xCE, 0xCE), (int) StateColor::Normal)); StaticBox::Create(parent, id, wxDefaultPosition, wxDefaultSize, 0); SetBackgroundColour(StaticBox::GetParentBackgroundColor(parent)); SetCursor(wxCursor(wxCURSOR_HAND)); m_tooltips[0] = _L("Simple settings"); m_tooltips[1] = _L("Advanced settings"); m_tooltips[2] = _L("Expert settings"); Bind(wxEVT_LEFT_DOWN, &ModeSwitchButton::mouseDown, this); Bind(wxEVT_LEFT_UP, &ModeSwitchButton::mouseReleased, this); Bind(wxEVT_LEFT_DCLICK, &ModeSwitchButton::mouseDown, this); Bind(wxEVT_MOUSE_CAPTURE_LOST, &ModeSwitchButton::mouseCaptureLost, this); Rescale(); } void ModeSwitchButton::SetSelection(int selection) { m_selection = std::clamp(selection, 0, 2); update_tooltip(); Refresh(); } void ModeSwitchButton::SelectAndNotify(int selection) { if (!IsEnabled()) return; SetSelection(selection); Slic3r::GUI::wxGetApp().save_mode(m_selection); } void ModeSwitchButton::Rescale() { const wxSize button_size = FromDIP(wxSize(48, 20)); SetMinSize(button_size); SetMaxSize(button_size); SetSize(button_size); SetCornerRadius(button_size.y / 2.0); Refresh(); } bool ModeSwitchButton::Enable(bool enable /* = true */) { const bool changed = StaticBox::Enable(enable); if (changed) Refresh(); return changed; } void ModeSwitchButton::doRender(wxDC& dc) { dc.SetPen(*wxTRANSPARENT_PEN); dc.SetBrush(wxBrush(GetBackgroundColour())); dc.DrawRectangle(GetClientRect()); const wxRect bounds = GetClientRect().Deflate(1); if (bounds.width <= 0 || bounds.height <= 0) return; const int states = state_handler.states(); const bool hovered = (states & StateHandler::Hovered) != 0; const bool focused = (states & StateHandler::Focused) != 0; const bool disabled = !IsEnabled(); const wxColour track_fill = disabled ? wxColour(0xD0, 0xD0, 0xD4) : m_pressed ? wxColour(0x5A, 0x5D, 0x64) : wxColour(0x66, 0x69, 0x70); const wxColour track_border = disabled ? wxColour(0xDD, 0xDD, 0xE0) : focused ? wxColour("#009688") : hovered ? wxColour(0x7A, 0x7D, 0x84) : wxColour(0x75, 0x78, 0x7F); const wxColour active_fill = disabled ? wxColour(0x9E, 0xBE, 0xB9) : m_pressed ? wxColour(0x00877B) : wxColour("#009688"); const wxColour active_dot = disabled ? wxColour(0xEC, 0xF4, 0xF2) : wxColour(0xB7, 0xEB, 0xE3); const wxColour inactive_dot = disabled ? wxColour(0xF2, 0xF2, 0xF4) : wxColour(0xB5, 0xB7, 0xBD); const wxColour thumb_fill = disabled ? wxColour(0xFA, 0xFA, 0xFA) : *wxWHITE; const wxColour thumb_border = disabled ? wxColour(0xE7, 0xE7, 0xEA) : wxColour(0xDD, 0xDF, 0xE3); dc.SetPen(wxPen(track_border, 1)); dc.SetBrush(wxBrush(track_fill)); dc.DrawRoundedRectangle(bounds, bounds.height / 2.0); const wxRect thumb = thumb_rect_for(m_selection); const int fill_right = std::min(bounds.GetRight(), thumb.GetX() + thumb.GetWidth() / 2 + FromDIP(2)); wxRect active(bounds.x, bounds.y, fill_right - bounds.x + 1, bounds.height); dc.SetPen(*wxTRANSPARENT_PEN); dc.SetBrush(wxBrush(active_fill)); dc.DrawRoundedRectangle(active, bounds.height / 2.0); const int dot_radius = std::max(FromDIP(1), thumb.height / 7); for (int idx = 0; idx < 3; ++idx) { if (idx == m_selection) continue; const wxRect slot = thumb_rect_for(idx); const wxPoint center(slot.GetX() + slot.GetWidth() / 2, slot.GetY() + slot.GetHeight() / 2); dc.SetBrush(wxBrush(idx < m_selection ? active_dot : inactive_dot)); dc.DrawCircle(center, dot_radius); } dc.SetPen(wxPen(thumb_border, 1)); dc.SetBrush(wxBrush(thumb_fill)); dc.DrawRoundedRectangle(thumb, thumb.height / 2.0); } void ModeSwitchButton::mouseDown(wxMouseEvent& event) { if (!IsEnabled()) { event.Skip(); return; } m_pressed = true; if (!HasCapture()) CaptureMouse(); Refresh(); event.Skip(); } void ModeSwitchButton::mouseReleased(wxMouseEvent& event) { if (m_pressed) { m_pressed = false; if (HasCapture()) ReleaseMouse(); if (GetClientRect().Contains(event.GetPosition())) SelectAndNotify(hit_test_selection(event.GetPosition())); Refresh(); } event.Skip(); } void ModeSwitchButton::mouseCaptureLost(wxMouseCaptureLostEvent& event) { m_pressed = false; Refresh(); event.Skip(); } int ModeSwitchButton::hit_test_selection(const wxPoint& point) const { const int width = std::max(1, GetClientSize().x); const int x = std::clamp(point.x, 0, width - 1); return std::clamp((x * 3) / width, 0, 2); } wxRect ModeSwitchButton::thumb_rect_for(int selection) const { const wxRect bounds = GetClientRect().Deflate(3); const int thumb_diameter = std::max(FromDIP(10), bounds.height - FromDIP(2)); const int y = bounds.y + (bounds.height - thumb_diameter) / 2; const int travel = std::max(0, bounds.width - thumb_diameter); const int x = bounds.x + (travel * std::clamp(selection, 0, 2)) / 2; return wxRect(x, y, thumb_diameter, thumb_diameter); } void ModeSwitchButton::update_tooltip() { SetToolTip(m_tooltips[m_selection]); } SwitchBoard::SwitchBoard(wxWindow *parent, wxString leftL, wxString right, wxSize size) : wxWindow(parent, wxID_ANY, wxDefaultPosition, size) { #ifdef __WINDOWS__ SetDoubleBuffered(true); #endif //__WINDOWS__ SetBackgroundColour(*wxWHITE); leftLabel = leftL; rightLabel = right; SetMinSize(size); SetMaxSize(size); Bind(wxEVT_PAINT, &SwitchBoard::paintEvent, this); Bind(wxEVT_LEFT_DOWN, &SwitchBoard::on_left_down, this); Bind(wxEVT_ENTER_WINDOW, [this](auto &e) { SetCursor(wxCURSOR_HAND); }); Bind(wxEVT_LEAVE_WINDOW, [this](auto &e) { SetCursor(wxCURSOR_ARROW); }); } void SwitchBoard::updateState(wxString target) { if (target.empty()) { if (!switch_left && !switch_right) { return; } switch_left = false; switch_right = false; } else { if (target == "left") { if (switch_left && !switch_right) { return; } switch_left = true; switch_right = false; } else if (target == "right") { if (!switch_left && switch_right) { return; } switch_left = false; switch_right = true; } } Refresh(); } void SwitchBoard::paintEvent(wxPaintEvent &evt) { wxPaintDC dc(this); render(dc); } void SwitchBoard::render(wxDC &dc) { #ifdef __WXMSW__ wxSize size = GetSize(); wxMemoryDC memdc; wxBitmap bmp(size.x, size.y); memdc.SelectObject(bmp); memdc.Blit({0, 0}, size, &dc, {0, 0}); { wxGCDC dc2(memdc); doRender(dc2); } memdc.SelectObject(wxNullBitmap); dc.DrawBitmap(bmp, 0, 0); #else doRender(dc); #endif } void SwitchBoard::doRender(wxDC &dc) { wxColour disable_color = wxColour(0xCECECE); dc.SetPen(*wxTRANSPARENT_PEN); if (is_enable) {dc.SetBrush(wxBrush(0xeeeeee)); } else {dc.SetBrush(disable_color);} dc.DrawRoundedRectangle(0, 0, GetSize().x, GetSize().y, 8); /*left*/ if (switch_left) { is_enable ? dc.SetBrush(wxBrush(wxColour(0, 150, 136))) : dc.SetBrush(disable_color); dc.DrawRoundedRectangle(0, 0, GetSize().x / 2, GetSize().y, 8); } if (switch_left) { dc.SetTextForeground(*wxWHITE); } else { dc.SetTextForeground(0x333333); } dc.SetFont(::Label::Body_13); Slic3r::GUI::WxFontUtils::get_suitable_font_size(0.6 * GetSize().GetHeight(), dc); auto left_txt_size = dc.GetTextExtent(leftLabel); dc.DrawText(leftLabel, wxPoint((GetSize().x / 2 - left_txt_size.x) / 2, (GetSize().y - left_txt_size.y) / 2)); /*right*/ if (switch_right) { if (is_enable) {dc.SetBrush(wxBrush(wxColour(0, 150, 136))); } else {dc.SetBrush(disable_color);} dc.DrawRoundedRectangle(GetSize().x / 2, 0, GetSize().x / 2, GetSize().y, 8); } auto right_txt_size = dc.GetTextExtent(rightLabel); if (switch_right) { dc.SetTextForeground(*wxWHITE); } else { dc.SetTextForeground(0x333333); } dc.DrawText(rightLabel, wxPoint((GetSize().x / 2 - right_txt_size.x) / 2 + GetSize().x / 2, (GetSize().y - right_txt_size.y) / 2)); } void SwitchBoard::on_left_down(wxMouseEvent &evt) { if (!is_enable) { return; } int index = -1; auto pos = ClientToScreen(evt.GetPosition()); auto rect = ClientToScreen(wxPoint(0, 0)); if (pos.x > 0 && pos.x < rect.x + GetSize().x / 2) { switch_left = true; switch_right = false; index = 1; } else { switch_left = false; switch_right = true; index = 0; } if (auto_disable_when_switch) { is_enable = false;// make it disable while switching } Refresh(); wxCommandEvent event(wxCUSTOMEVT_SWITCH_POS); event.SetInt(index); wxPostEvent(this, event); } bool SwitchBoard::Enable(bool enable /* = true */) { if (is_enable == enable) { return false; } is_enable = enable; Refresh(); return true; }