mirror of
https://github.com/SoftFever/OrcaSlicer.git
synced 2025-08-30 05:21:51 -06:00

* FIX: the logic of buried points that were not buried JIRA: none Signed-off-by: Kunlong Ma <kunlong.ma@bambulab.com> Change-Id: Id95174659c5fce7feba409eb5e14916608745fa4 * ci: update network module based on commit bc7ca98 Change-Id: I923526f0bf9ce5a288144fa1f9b0f2fc640f41b7 * Fix Firefox Co-authored-by: hadess <hadess@hadess.net> * FIX: cali: custom created filament from AMS displayed as incompatible jira: new remove the condition: is_system Change-Id: Ib1366966bbdbe01bc9e2483d9914d270ebefa976 * FIX: duplicated items in comboBox at calibration completed page jira: new Change-Id: I4749a2206df16c438e0d3098e36274b2a20f313e * ENH:update support for P1S plus jira:[for p1s plus] Change-Id: Id577d4e94e2162cb0045d261dfaa5f396ecded2f * ENH: CLI: add mk information support JIRA: no jira Change-Id: Idd89b143d439de50d9f52eb8aec95b262d66875d * ENH:calibration support p1p plus jira:[plus] Change-Id: Ia290d3a8a8b9adaac7a6ee26d9a8b5ea0c1b3aee * FIX: add log for base_id and filament_id github: #3087 Change-Id: Iebfbd0f224fce49f33fc81c71e6108f6e3abb5ff * FIX: sync whole preset vendor directory Change-Id: I191dbe979a87ff35d38cab1149b7975664344838 Jira: STUDIO-5534 (cherry picked from commit 628866608116336453804aa1217dd55db04d47ad) * FIX: use t_utc for debug only Change-Id: Ia05d8969d4de3dd38908980d6e17a3ebb11ca279 Github 3045 Change-Id: I77935df53bbf2772b1146e5c330c537165a3a2e6 * FIX:make sort_volumes right Jira: STUDIO-5645 Change-Id: If324c9115bfaaf0c1b7b4be7c7ee96ba6b8ac890 * ENH:keep an unload logic jira:[for unload] Change-Id: Id30ec71ffa5b2dac89346ea47ca48a62479e3ab1 * FIX: several problems with mesh boolean 1. Cut with multiple volumes are OK now. 2. Close mesh boolean fail error with new object or open object 3. Fix wrong name and config of boolean resulting object github: #3118 jira: none Change-Id: If2c9dbfb36cbdfe4917a2371217923891bb7909c (cherry picked from commit 982c0ecb92cf7c2b5ae5972ab900a6b10e7dda50) * NEW:limit the length of project name jira:[project name] Change-Id: I955620f7073b3b7fda280d1118524f561d047751 * ENH:adjusting the warning level of timelpase jira:[STUDIO-5662] Change-Id: I4902b22d316f5e09a97a62c88b8a98e55c405434 * FIX: 3mf specification: change namespace form slic3rpe to BambuStudio Jira: XXXX Change-Id: Id705affc875ef23fdf2ac6d79f0cb0aafc4f7050 * NEW: Open MakerWorld With BambuStudio GetParam JIRA: none Change-Id: I0d65b364f1cd2d634a88882ab072c3e61ea89167 (cherry picked from commit 8eaf45e5359439a7c796fd79876c86775abcf48e) * FIX: Filament issue generated when creating a printer Jira: XXXX Change-Id: I976770b69b47641bd54aa7a9c56fba7f58d1ab68 (cherry picked from commit ba42188b93c58b3954234d72acdd9769a68e3d3c) * FIX: Blank page appears when editing presets Jira: 5563 Change-Id: I4c49e05515b1beff55991e92f8079c4499c27eab (cherry picked from commit e86517d290f4cd0765a230d811b0ddf2c9f34c17) * FIX: context menu didn't update UI jira: STUDIO-5691 Change-Id: Ia66b8623d832eba805aff5320941233a68ff258b * FIX: crash of "filling bed" "get_arrange_settings() const" gets trapped in infinite recursive calling. Now we delete this function. jira: STUDIO-5688 Change-Id: Ia39974734bb37b2a2f06b5bf78185f01be726872 * FIX: boolean hangs in the middle of color painting Can't do splits in combine_mesh_fff, as do_boolean of mcut will split meshes. jira: STUDIO-5693 Change-Id: Idddb7d20dd7ca386c39ddd3d87d9defc3136aa5d (cherry picked from commit 6c67d015941458e37faaf0015b6509b5a0eadc0e) * Fix: Fix a number of compilation problems issues found when using gcc version 13.2.0 (GCC) in a Flatpak sandbox github : https://github.com/bambulab/BambuStudio/issues/3074 github pull request: https://github.com/bambulab/BambuStudio/pull/3096 Change-Id: I08aeac593eb1ce7675894df72e8489200bae713d (cherry picked from commit 069d133d66bfa682de4a860e379d5dc16b3d907c) * fix: macos icns issue when icon was not attached github pull request:https://github.com/bambulab/BambuStudio/pull/3116 Change-Id: I49072ad49f3af7669a6d307c791594ade210da50 (cherry picked from commit c977e5582e3a30ad16dd267810037423aad9a53c) * FIX: Add flush_length for change_filament_gcode Change-Id: I30f4b97d3d61c2a57f0e92f157cbd31c38aa7265 Jira: XXXX (cherry picked from commit 92eb2bac977a0c4095b316cbbc6580fb5228b710) * FIX: edit preset dialog can't close on mac Jira: 5696 Change-Id: Ib33dfd07cc588ddd3805e3490a4d8c36dcd890ac * ENH: add dev_ota_version in ssdp JIRA: STUDIO-5740 Change-Id: Ic80e6d4b9bf82813fdc4a76604a3d36213d12b03 Signed-off-by: Stone Li <stone.li@bambulab.com> * NEW:Adapt to multicolour and gradient colour JIRA:xxxx Change-Id: I8084cab603d5681cbcaf2d6f5e0d7ad5419cb2af * NEW:Adaptation of semi transparent materials JIRA: XXXX Change-Id: Ie32d8ce67c37b85eb6d7d6594cb514a696307e68 * FIX: disable flush options if prime tower is unchecked jira: STUDIO-5639 Change-Id: I25081584d430bc7d062e14bcc2cdbf7522cf9d99 * ENH: refine GetVersion for HMS query JIRA: STUDIO-5763 Change-Id: Ia3ccc07d79cc0736eb12e9782da50211abb74772 Signed-off-by: Stone Li <stone.li@bambulab.com> * FIX: Prefer old selection when sync AMS not compatible Change-Id: I6b18db51887132a997cf78d70fff9a92e23bc44a Jira: STUDIO-5416 (cherry picked from commit 077fae29823cf4f3071d408b1b40f55ee0cb33c6) * FIX: The flushing was not auto-calc when sync ams list JIRA: STUDIO-5551 1. flushing volume auto-calc when sync ams list 2. flushing volume takes the larger calculation value when filament has multi-colors Signed-off-by: Kunlong Ma <kunlong.ma@bambulab.com> Change-Id: I72e6f9780ea56103a44c2da6068440a4615c254d * FIX:fixed invalid links jira:[fixed link] Change-Id: I036a38b6e8e94da762f93805bd7be706538771fe * FIX: Prompt to delete problematic presets Jira: XXXX Change-Id: Ic43f7bb782794d7ab0b6acbffbb5d73e94f6ed73 * FIX:fixed incorrect HMS content jira:[STUDIO-5818] Change-Id: Ia2896d6f0ab1ffedbc850e54168acece8e47bdbb * FIX:external transparent material display error JIRA: STUDIO-5845 Change-Id: I0a4f05ac5d5c0ac49d85a704ee65a7221c5f1e1d * FIX: [5846] Custom Filament Page show System Filament Simultaneously solve: When downloading Preset from the cloud, the filament_id of the preset in m_preset is null. Jira: 5846 Change-Id: I6ba1b46fe92e345614b6a4af3fffa87d81fa2456 * FIX:A1 and p1 series do not support custom materials JIRA:XXXX Change-Id: Ib0459273d1f9a7152a5563757204634a8d0cd6f5 * FIX: exception when comparing profiles jira:[NEW] Signed-off-by: XunZhangBambu <xun.zhang@bambulab.com> Change-Id: I946b5fcd35f779d271df2b3de731fdcada5aab29 (cherry picked from commit 00e739570812e5c4be3e0f7702ce8c72c0f9e72b) * FIX: hide_id_middle_string Change-Id: I28f32ec526b443d31d7992971b80ab1cb737deb6 Github: STUDIO-5825 * ENH: modify some logs level JIRA: STUDIO-5958 Change-Id: I5a8592dfb8ffa9a81952535cb30944f867aa0e22 Signed-off-by: Stone Li <stone.li@bambulab.com> * NEW:build plate marker detect Change-Id: I70f03efea688bb6ce71c3f5990bb3c50605ab184 * FIX: Studio UI Freeze when saving user preset github: #3335 Change-Id: Idaf53f673a3e46408826c06bdde2c592395d358b * update bbl plugin version * fix build errors * update bbl profiles * update color --------- Signed-off-by: Kunlong Ma <kunlong.ma@bambulab.com> Signed-off-by: Stone Li <stone.li@bambulab.com> Co-authored-by: Kunlong Ma <kunlong.ma@bambulab.com> Co-authored-by: gerrit <gerrit@bambulab.com> Co-authored-by: liz.li <liz.li@bambulab.com> Co-authored-by: tao wang <tao.wang@bambulab.com> Co-authored-by: lane.wei <lane.wei@bambulab.com> Co-authored-by: maosheng.wei <maosheng.wei@bambulab.com> Co-authored-by: chunmao.guo <chunmao.guo@bambulab.com> Co-authored-by: zhou.xu <zhou.xu@bambulab.com> Co-authored-by: Arthur <arthur.tang@bambulab.com> Co-authored-by: Bastien Nocera <hadess@hadess.net> Co-authored-by: zhimin.zeng <zhimin.zeng@bambulab.com> Co-authored-by: hu.wang <hu.wang@bambulab.com> Co-authored-by: Stone Li <stone.li@bambulab.com> Co-authored-by: XunZhangBambu <xun.zhang@bambulab.com>
1198 lines
37 KiB
C++
1198 lines
37 KiB
C++
#include "wxExtensions.hpp"
|
|
|
|
#include <stdexcept>
|
|
#include <cmath>
|
|
|
|
#include <wx/sizer.h>
|
|
|
|
#include <boost/algorithm/string/replace.hpp>
|
|
|
|
#include "GUI.hpp"
|
|
#include "GUI_App.hpp"
|
|
#include "GUI_ObjectList.hpp"
|
|
#include "I18N.hpp"
|
|
#include "GUI_Utils.hpp"
|
|
#include "Plater.hpp"
|
|
#include "../Utils/MacDarkMode.hpp"
|
|
#include "BitmapComboBox.hpp"
|
|
#include "Widgets/StaticBox.hpp"
|
|
#include "Widgets/Label.hpp"
|
|
|
|
#ifndef __linux__
|
|
// msw_menuitem_bitmaps is used for MSW and OSX
|
|
static std::map<int, std::string> msw_menuitem_bitmaps;
|
|
#ifdef __WXMSW__
|
|
void msw_rescale_menu(wxMenu* menu)
|
|
{
|
|
struct update_icons {
|
|
static void run(wxMenuItem* item) {
|
|
const auto it = msw_menuitem_bitmaps.find(item->GetId());
|
|
if (it != msw_menuitem_bitmaps.end()) {
|
|
const wxBitmap& item_icon = create_menu_bitmap(it->second);
|
|
if (item_icon.IsOk())
|
|
item->SetBitmap(item_icon);
|
|
}
|
|
if (item->IsSubMenu())
|
|
for (wxMenuItem *sub_item : item->GetSubMenu()->GetMenuItems())
|
|
update_icons::run(sub_item);
|
|
}
|
|
};
|
|
|
|
for (wxMenuItem *item : menu->GetMenuItems())
|
|
update_icons::run(item);
|
|
}
|
|
#endif /* __WXMSW__ */
|
|
#endif /* no __WXGTK__ */
|
|
|
|
void enable_menu_item(wxUpdateUIEvent& evt, std::function<bool()> const cb_condition, wxMenuItem* item, wxWindow* win)
|
|
{
|
|
const bool enable = cb_condition();
|
|
evt.Enable(enable);
|
|
|
|
#ifdef __WXOSX__
|
|
const auto it = msw_menuitem_bitmaps.find(item->GetId());
|
|
if (it != msw_menuitem_bitmaps.end())
|
|
{
|
|
const wxBitmap& item_icon = create_scaled_bitmap(it->second, win, 16, !enable);
|
|
if (item_icon.IsOk())
|
|
item->SetBitmap(item_icon);
|
|
}
|
|
#endif // __WXOSX__
|
|
}
|
|
|
|
wxMenuItem* append_menu_item(wxMenu* menu, int id, const wxString& string, const wxString& description,
|
|
std::function<void(wxCommandEvent& event)> cb, const wxBitmap& icon, wxEvtHandler* event_handler,
|
|
std::function<bool()> const cb_condition, wxWindow* parent, int insert_pos/* = wxNOT_FOUND*/)
|
|
{
|
|
if (id == wxID_ANY)
|
|
id = wxNewId();
|
|
|
|
auto *item = new wxMenuItem(menu, id, string, description);
|
|
if (icon.IsOk()) {
|
|
item->SetBitmap(icon);
|
|
}
|
|
if (insert_pos == wxNOT_FOUND)
|
|
menu->Append(item);
|
|
else
|
|
menu->Insert(insert_pos, item);
|
|
|
|
#ifdef __WXMSW__
|
|
if (event_handler != nullptr && event_handler != menu)
|
|
event_handler->Bind(wxEVT_MENU, cb, id);
|
|
else
|
|
#endif // __WXMSW__
|
|
menu->Bind(wxEVT_MENU, cb, id);
|
|
|
|
if (parent) {
|
|
parent->Bind(wxEVT_UPDATE_UI, [cb_condition, item, parent](wxUpdateUIEvent& evt) {
|
|
enable_menu_item(evt, cb_condition, item, parent); }, id);
|
|
}
|
|
|
|
return item;
|
|
}
|
|
|
|
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,
|
|
std::function<bool()> const cb_condition, wxWindow* parent, int insert_pos/* = wxNOT_FOUND*/)
|
|
{
|
|
if (id == wxID_ANY)
|
|
id = wxNewId();
|
|
|
|
const wxBitmap& bmp = !icon.empty() ? create_menu_bitmap(icon) : wxNullBitmap; // FIXME: pass window ptr
|
|
//#ifdef __WXMSW__
|
|
#ifndef __WXGTK__
|
|
if (bmp.IsOk())
|
|
msw_menuitem_bitmaps[id] = icon;
|
|
#endif /* __WXMSW__ */
|
|
|
|
return append_menu_item(menu, id, string, description, cb, bmp, event_handler, cb_condition, parent, insert_pos);
|
|
}
|
|
|
|
wxMenuItem* append_submenu(wxMenu* menu, wxMenu* sub_menu, int id, const wxString& string, const wxString& description, const std::string& icon,
|
|
std::function<bool()> const cb_condition, wxWindow* parent)
|
|
{
|
|
if (id == wxID_ANY)
|
|
id = wxNewId();
|
|
|
|
wxMenuItem* item = new wxMenuItem(menu, id, string, description, wxITEM_NORMAL, sub_menu);
|
|
if (!icon.empty()) {
|
|
item->SetBitmap(create_menu_bitmap(icon)); // FIXME: pass window ptr
|
|
//#ifdef __WXMSW__
|
|
#ifndef __WXGTK__
|
|
msw_menuitem_bitmaps[id] = icon;
|
|
#endif /* __WXMSW__ */
|
|
}
|
|
|
|
menu->Append(item);
|
|
|
|
if (parent) {
|
|
parent->Bind(wxEVT_UPDATE_UI, [cb_condition, item, parent](wxUpdateUIEvent& evt) {
|
|
enable_menu_item(evt, cb_condition, item, parent); }, id);
|
|
}
|
|
|
|
return item;
|
|
}
|
|
|
|
wxMenuItem* append_menu_radio_item(wxMenu* menu, int id, const wxString& string, const wxString& description,
|
|
std::function<void(wxCommandEvent& event)> cb, wxEvtHandler* event_handler)
|
|
{
|
|
if (id == wxID_ANY)
|
|
id = wxNewId();
|
|
|
|
wxMenuItem* item = menu->AppendRadioItem(id, string, description);
|
|
|
|
#ifdef __WXMSW__
|
|
if (event_handler != nullptr && event_handler != menu)
|
|
event_handler->Bind(wxEVT_MENU, cb, id);
|
|
else
|
|
#endif // __WXMSW__
|
|
menu->Bind(wxEVT_MENU, cb, id);
|
|
|
|
return item;
|
|
}
|
|
|
|
wxMenuItem* append_menu_check_item(wxMenu* menu, int id, const wxString& string, const wxString& description,
|
|
std::function<void(wxCommandEvent & event)> cb, wxEvtHandler* event_handler,
|
|
std::function<bool()> const enable_condition, std::function<bool()> const check_condition, wxWindow* parent)
|
|
{
|
|
if (id == wxID_ANY)
|
|
id = wxNewId();
|
|
|
|
wxMenuItem* item = menu->AppendCheckItem(id, string, description);
|
|
|
|
#ifdef __WXMSW__
|
|
if (event_handler != nullptr && event_handler != menu)
|
|
event_handler->Bind(wxEVT_MENU, cb, id);
|
|
else
|
|
#endif // __WXMSW__
|
|
menu->Bind(wxEVT_MENU, cb, id);
|
|
|
|
if (parent)
|
|
parent->Bind(wxEVT_UPDATE_UI, [enable_condition, check_condition](wxUpdateUIEvent& evt)
|
|
{
|
|
evt.Enable(enable_condition());
|
|
evt.Check(check_condition());
|
|
}, id);
|
|
|
|
return item;
|
|
}
|
|
|
|
const unsigned int wxCheckListBoxComboPopup::DefaultWidth = 200;
|
|
const unsigned int wxCheckListBoxComboPopup::DefaultHeight = 200;
|
|
|
|
bool wxCheckListBoxComboPopup::Create(wxWindow* parent)
|
|
{
|
|
return wxCheckListBox::Create(parent, wxID_HIGHEST + 1, wxPoint(0, 0));
|
|
}
|
|
|
|
wxWindow* wxCheckListBoxComboPopup::GetControl()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
void wxCheckListBoxComboPopup::SetStringValue(const wxString& value)
|
|
{
|
|
m_text = value;
|
|
}
|
|
|
|
wxString wxCheckListBoxComboPopup::GetStringValue() const
|
|
{
|
|
return m_text;
|
|
}
|
|
|
|
wxSize wxCheckListBoxComboPopup::GetAdjustedSize(int minWidth, int prefHeight, int maxHeight)
|
|
{
|
|
// set width dinamically in dependence of items text
|
|
// and set height dinamically in dependence of items count
|
|
|
|
wxComboCtrl* cmb = GetComboCtrl();
|
|
if (cmb != nullptr) {
|
|
wxSize size = GetComboCtrl()->GetSize();
|
|
|
|
unsigned int count = GetCount();
|
|
if (count > 0) {
|
|
int max_width = size.x;
|
|
for (unsigned int i = 0; i < count; ++i) {
|
|
max_width = std::max(max_width, 60 + GetTextExtent(GetString(i)).x);
|
|
}
|
|
size.SetWidth(max_width);
|
|
size.SetHeight(count * cmb->GetCharHeight());
|
|
}
|
|
else
|
|
size.SetHeight(DefaultHeight);
|
|
|
|
return size;
|
|
}
|
|
else
|
|
return wxSize(DefaultWidth, DefaultHeight);
|
|
}
|
|
|
|
void wxCheckListBoxComboPopup::OnKeyEvent(wxKeyEvent& evt)
|
|
{
|
|
// filters out all the keys which are not working properly
|
|
switch (evt.GetKeyCode())
|
|
{
|
|
case WXK_LEFT:
|
|
case WXK_UP:
|
|
case WXK_RIGHT:
|
|
case WXK_DOWN:
|
|
case WXK_PAGEUP:
|
|
case WXK_PAGEDOWN:
|
|
case WXK_END:
|
|
case WXK_HOME:
|
|
case WXK_NUMPAD_LEFT:
|
|
case WXK_NUMPAD_UP:
|
|
case WXK_NUMPAD_RIGHT:
|
|
case WXK_NUMPAD_DOWN:
|
|
case WXK_NUMPAD_PAGEUP:
|
|
case WXK_NUMPAD_PAGEDOWN:
|
|
case WXK_NUMPAD_END:
|
|
case WXK_NUMPAD_HOME:
|
|
{
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
evt.Skip();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void wxCheckListBoxComboPopup::OnCheckListBox(wxCommandEvent& evt)
|
|
{
|
|
// forwards the checklistbox event to the owner wxComboCtrl
|
|
|
|
if (m_check_box_events_status == OnCheckListBoxFunction::FreeToProceed )
|
|
{
|
|
wxComboCtrl* cmb = GetComboCtrl();
|
|
if (cmb != nullptr) {
|
|
wxCommandEvent event(wxEVT_CHECKLISTBOX, cmb->GetId());
|
|
event.SetEventObject(cmb);
|
|
cmb->ProcessWindowEvent(event);
|
|
}
|
|
}
|
|
|
|
evt.Skip();
|
|
|
|
#ifndef _WIN32 // events are sent differently on OSX+Linux vs Win (more description in header file)
|
|
if ( m_check_box_events_status == OnCheckListBoxFunction::RefuseToProceed )
|
|
// this happens if the event was resent by OnListBoxSelection - next call to OnListBoxSelection is due to user clicking the text, so the function should
|
|
// explicitly change the state on the checkbox
|
|
m_check_box_events_status = OnCheckListBoxFunction::WasRefusedLastTime;
|
|
else
|
|
// if the user clicked the checkbox square, this event was sent before OnListBoxSelection was called, so we don't want it to resend it
|
|
m_check_box_events_status = OnCheckListBoxFunction::RefuseToProceed;
|
|
#endif
|
|
}
|
|
|
|
void wxCheckListBoxComboPopup::OnListBoxSelection(wxCommandEvent& evt)
|
|
{
|
|
// transforms list box item selection event into checklistbox item toggle event
|
|
|
|
int selId = GetSelection();
|
|
if (selId != wxNOT_FOUND)
|
|
{
|
|
#ifndef _WIN32
|
|
if (m_check_box_events_status == OnCheckListBoxFunction::RefuseToProceed)
|
|
#endif
|
|
Check((unsigned int)selId, !IsChecked((unsigned int)selId));
|
|
|
|
m_check_box_events_status = OnCheckListBoxFunction::FreeToProceed; // so the checkbox reacts to square-click the next time
|
|
|
|
SetSelection(wxNOT_FOUND);
|
|
wxCommandEvent event(wxEVT_CHECKLISTBOX, GetId());
|
|
event.SetInt(selId);
|
|
event.SetEventObject(this);
|
|
ProcessEvent(event);
|
|
}
|
|
}
|
|
|
|
|
|
// *** wxDataViewTreeCtrlComboPopup ***
|
|
|
|
const unsigned int wxDataViewTreeCtrlComboPopup::DefaultWidth = 270;
|
|
const unsigned int wxDataViewTreeCtrlComboPopup::DefaultHeight = 200;
|
|
const unsigned int wxDataViewTreeCtrlComboPopup::DefaultItemHeight = 22;
|
|
|
|
bool wxDataViewTreeCtrlComboPopup::Create(wxWindow* parent)
|
|
{
|
|
return wxDataViewTreeCtrl::Create(parent, wxID_ANY/*HIGHEST + 1*/, wxPoint(0, 0), wxDefaultSize/*wxSize(270, -1)*/, wxDV_NO_HEADER);
|
|
}
|
|
/*
|
|
wxSize wxDataViewTreeCtrlComboPopup::GetAdjustedSize(int minWidth, int prefHeight, int maxHeight)
|
|
{
|
|
// matches owner wxComboCtrl's width
|
|
// and sets height dinamically in dependence of contained items count
|
|
wxComboCtrl* cmb = GetComboCtrl();
|
|
if (cmb != nullptr)
|
|
{
|
|
wxSize size = GetComboCtrl()->GetSize();
|
|
if (m_cnt_open_items > 0)
|
|
size.SetHeight(m_cnt_open_items * DefaultItemHeight);
|
|
else
|
|
size.SetHeight(DefaultHeight);
|
|
|
|
return size;
|
|
}
|
|
else
|
|
return wxSize(DefaultWidth, DefaultHeight);
|
|
}
|
|
*/
|
|
void wxDataViewTreeCtrlComboPopup::OnKeyEvent(wxKeyEvent& evt)
|
|
{
|
|
// filters out all the keys which are not working properly
|
|
if (evt.GetKeyCode() == WXK_UP)
|
|
{
|
|
return;
|
|
}
|
|
else if (evt.GetKeyCode() == WXK_DOWN)
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
evt.Skip();
|
|
return;
|
|
}
|
|
}
|
|
|
|
void wxDataViewTreeCtrlComboPopup::OnDataViewTreeCtrlSelection(wxCommandEvent& evt)
|
|
{
|
|
wxComboCtrl* cmb = GetComboCtrl();
|
|
auto selected = GetItemText(GetSelection());
|
|
cmb->SetText(selected);
|
|
}
|
|
|
|
// edit tooltip : change Slic3r to SLIC3R_APP_KEY
|
|
// Temporary workaround for localization
|
|
void edit_tooltip(wxString& tooltip)
|
|
{
|
|
tooltip.Replace("Slic3r", SLIC3R_APP_KEY, true);
|
|
}
|
|
|
|
/* Function for rescale of buttons in Dialog under MSW if dpi is changed.
|
|
* btn_ids - vector of buttons identifiers
|
|
*/
|
|
void msw_buttons_rescale(wxDialog* dlg, const int em_unit, const std::vector<int>& btn_ids)
|
|
{
|
|
const wxSize& btn_size = wxSize(-1, int(2.5f * em_unit + 0.5f));
|
|
|
|
for (int btn_id : btn_ids) {
|
|
// There is a case [FirmwareDialog], when we have wxControl instead of wxButton
|
|
// so let casting everything to the wxControl
|
|
wxControl* btn = static_cast<wxControl*>(dlg->FindWindowById(btn_id, dlg));
|
|
if (btn)
|
|
btn->SetMinSize(btn_size);
|
|
}
|
|
}
|
|
|
|
/* Function for getting of em_unit value from correct parent.
|
|
* In most of cases it is m_em_unit value from GUI_App,
|
|
* but for DPIDialogs it's its own value.
|
|
* This value will be used to correct rescale after moving between
|
|
* Displays with different HDPI */
|
|
int em_unit(wxWindow* win)
|
|
{
|
|
if (win)
|
|
{
|
|
wxTopLevelWindow *toplevel = Slic3r::GUI::find_toplevel_parent(win);
|
|
Slic3r::GUI::DPIDialog* dlg = dynamic_cast<Slic3r::GUI::DPIDialog*>(toplevel);
|
|
if (dlg)
|
|
return dlg->em_unit();
|
|
Slic3r::GUI::DPIFrame* frame = dynamic_cast<Slic3r::GUI::DPIFrame*>(toplevel);
|
|
if (frame)
|
|
return frame->em_unit();
|
|
}
|
|
|
|
return Slic3r::GUI::wxGetApp().em_unit();
|
|
}
|
|
|
|
int mode_icon_px_size()
|
|
{
|
|
#ifdef __APPLE__
|
|
return 10;
|
|
#else
|
|
return 12;
|
|
#endif
|
|
}
|
|
|
|
wxBitmap create_menu_bitmap(const std::string& bmp_name)
|
|
{
|
|
return create_scaled_bitmap(bmp_name, nullptr, 16, false, "", true);
|
|
}
|
|
|
|
// win is used to get a correct em_unit value
|
|
// It's important for bitmaps of dialogs.
|
|
// if win == nullptr, em_unit value of MainFrame will be used
|
|
wxBitmap create_scaled_bitmap( const std::string& bmp_name_in,
|
|
wxWindow *win/* = nullptr*/,
|
|
const int px_cnt/* = 16*/,
|
|
const bool grayscale/* = false*/,
|
|
const std::string& new_color/* = std::string()*/, // color witch will used instead of orange
|
|
const bool menu_bitmap/* = false*/,
|
|
const bool resize/* = false*/,
|
|
const bool bitmap2/* = false*/,
|
|
const vector<std::string>& array_new_color/* = vector<std::string>*/)//used for semi transparent material)
|
|
{
|
|
static Slic3r::GUI::BitmapCache cache;
|
|
if (bitmap2) {
|
|
return create_scaled_bitmap2(bmp_name_in, cache, win, px_cnt, grayscale, resize, array_new_color);
|
|
}
|
|
unsigned int width = 0;
|
|
unsigned int height = (unsigned int) (win->FromDIP(px_cnt) + 0.5f);
|
|
|
|
std::string bmp_name = bmp_name_in;
|
|
boost::replace_last(bmp_name, ".png", "");
|
|
|
|
bool dark_mode =
|
|
#ifdef _WIN32
|
|
menu_bitmap ? Slic3r::GUI::check_dark_mode() :
|
|
#endif
|
|
Slic3r::GUI::wxGetApp().dark_mode();
|
|
|
|
// Try loading an SVG first, then PNG if SVG is not found:
|
|
wxBitmap *bmp = cache.load_svg(bmp_name, width, height, grayscale, dark_mode, new_color, resize ? em_unit(win) * 0.1f : 0.f);
|
|
if (bmp == nullptr) {
|
|
bmp = cache.load_png(bmp_name, width, height, grayscale, resize ? win->FromDIP(10) * 0.1f : 0.f);
|
|
}
|
|
|
|
if (bmp == nullptr) {
|
|
// Neither SVG nor PNG has been found, raise error
|
|
throw Slic3r::RuntimeError("Could not load bitmap: " + bmp_name);
|
|
}
|
|
|
|
return *bmp;
|
|
}
|
|
|
|
wxBitmap create_scaled_bitmap2(const std::string& bmp_name_in, Slic3r::GUI::BitmapCache& cache, wxWindow* win/* = nullptr*/ ,
|
|
const int px_cnt/* = 16*/, const bool grayscale/* = false*/ , const bool resize/* = false*/ ,
|
|
const vector<std::string>& array_new_color/* = vector<std::string>()*/) // color witch will used instead of orange
|
|
{
|
|
unsigned int width = 0;
|
|
unsigned int height = (unsigned int)(win->FromDIP(px_cnt) + 0.5f);
|
|
|
|
std::string bmp_name = bmp_name_in;
|
|
boost::replace_last(bmp_name, ".png", "");
|
|
|
|
wxBitmap* bmp = cache.load_svg2(bmp_name, width, height, grayscale, false, array_new_color, resize ? em_unit(win) * 0.1f : 0.f);
|
|
if (bmp == nullptr) {
|
|
// No SVG found
|
|
throw Slic3r::RuntimeError("Could not load bitmap: " + bmp_name);
|
|
}
|
|
return *bmp;
|
|
}
|
|
|
|
|
|
wxBitmap* get_default_extruder_color_icon(bool thin_icon/* = false*/)
|
|
{
|
|
static Slic3r::GUI::BitmapCache bmp_cache;
|
|
|
|
const double em = Slic3r::GUI::wxGetApp().em_unit();
|
|
const int icon_width = lround((thin_icon ? 2 : 4.5) * em);
|
|
const int icon_height = lround(2 * em);
|
|
bool dark_mode = Slic3r::GUI::wxGetApp().dark_mode();
|
|
|
|
wxClientDC cdc((wxWindow*)Slic3r::GUI::wxGetApp().mainframe);
|
|
wxMemoryDC dc(&cdc);
|
|
dc.SetFont(::Label::Body_12);
|
|
|
|
wxString label = _L("default");
|
|
std::string bitmap_key = std::string("default_color") + "-h" + std::to_string(icon_height) + "-w" + std::to_string(icon_width)
|
|
+ "-i" + label.ToStdString();
|
|
|
|
wxBitmap* bitmap = bmp_cache.find(bitmap_key);
|
|
if (bitmap == nullptr) {
|
|
// Paint the color icon.
|
|
//Slic3r::GUI::BitmapCache::parse_color(color, rgb);
|
|
// there is no neede to scale created solid bitmap
|
|
wxColor clr(255, 255, 255, 0);
|
|
bitmap = bmp_cache.insert(bitmap_key, wxBitmap(icon_width, icon_height));
|
|
dc.SelectObject(*bitmap);
|
|
dc.SetBackground(wxBrush(clr));
|
|
dc.Clear();
|
|
dc.SetBrush(wxBrush(clr));
|
|
dc.SetPen(*wxGREY_PEN);
|
|
auto size = dc.GetTextExtent(wxString(label));
|
|
dc.SetTextForeground(clr.GetLuminance() < 0.51 ? *wxWHITE : *wxBLACK);
|
|
dc.DrawText(label, (icon_width - size.x) / 2, (icon_height - size.y) / 2);
|
|
dc.SelectObject(wxNullBitmap);
|
|
}
|
|
|
|
return bitmap;
|
|
}
|
|
|
|
std::vector<wxBitmap*> get_extruder_color_icons(bool thin_icon/* = false*/)
|
|
{
|
|
// Create the bitmap with color bars.
|
|
std::vector<wxBitmap*> bmps;
|
|
std::vector<std::string> colors = Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config();
|
|
|
|
if (colors.empty())
|
|
return bmps;
|
|
|
|
/* It's supposed that standard size of an icon is 36px*16px for 100% scaled display.
|
|
* So set sizes for solid_colored icons used for filament preset
|
|
* and scale them in respect to em_unit value
|
|
*/
|
|
const double em = Slic3r::GUI::wxGetApp().em_unit();
|
|
const int icon_width = lround((thin_icon ? 2 : 4.4) * em);
|
|
const int icon_height = lround(2 * em);
|
|
|
|
int index = 0;
|
|
for (const std::string &color : colors)
|
|
{
|
|
auto label = std::to_string(++index);
|
|
bmps.push_back(get_extruder_color_icon(color, label, icon_width, icon_height));
|
|
}
|
|
|
|
return bmps;
|
|
}
|
|
|
|
wxBitmap *get_extruder_color_icon(std::string color, std::string label, int icon_width, int icon_height)
|
|
{
|
|
static Slic3r::GUI::BitmapCache bmp_cache;
|
|
|
|
std::string bitmap_key = color + "-h" + std::to_string(icon_height) + "-w" + std::to_string(icon_width) + "-i" + label;
|
|
|
|
wxBitmap *bitmap = bmp_cache.find(bitmap_key);
|
|
if (bitmap == nullptr) {
|
|
// Paint the color icon.
|
|
// Slic3r::GUI::BitmapCache::parse_color(color, rgb);
|
|
// there is no neede to scale created solid bitmap
|
|
wxColor clr(color);
|
|
bitmap = bmp_cache.insert(bitmap_key, wxBitmap(icon_width, icon_height));
|
|
#ifndef __WXMSW__
|
|
wxMemoryDC dc;
|
|
#else
|
|
wxClientDC cdc((wxWindow *) Slic3r::GUI::wxGetApp().mainframe);
|
|
wxMemoryDC dc(&cdc);
|
|
#endif
|
|
dc.SetFont(::Label::Body_12);
|
|
dc.SelectObject(*bitmap);
|
|
if (clr.Alpha() == 0) {
|
|
int size = icon_height * 2;
|
|
static wxBitmap transparent = *Slic3r::GUI::BitmapCache().load_svg("transparent", size, size);
|
|
if (transparent.GetHeight() != size) transparent = *Slic3r::GUI::BitmapCache().load_svg("transparent", size, size);
|
|
wxPoint pt(0, 0);
|
|
while (pt.x < icon_width) {
|
|
dc.DrawBitmap(transparent, pt);
|
|
pt.x += size;
|
|
}
|
|
clr.SetRGB(0xffffff); // for text color
|
|
dc.SetBrush(*wxTRANSPARENT_BRUSH);
|
|
} else {
|
|
dc.SetBackground(wxBrush(clr));
|
|
dc.Clear();
|
|
dc.SetBrush(wxBrush(clr));
|
|
}
|
|
if (clr.Red() > 224 && clr.Blue() > 224 && clr.Green() > 224) {
|
|
dc.SetPen(*wxGREY_PEN);
|
|
dc.DrawRectangle(0, 0, icon_width, icon_height);
|
|
}
|
|
auto size = dc.GetTextExtent(wxString(label));
|
|
dc.SetTextForeground(clr.GetLuminance() < 0.51 ? *wxWHITE : *wxBLACK);
|
|
dc.DrawText(label, (icon_width - size.x) / 2, (icon_height - size.y) / 2);
|
|
dc.SelectObject(wxNullBitmap);
|
|
}
|
|
return bitmap;
|
|
}
|
|
|
|
|
|
void apply_extruder_selector(Slic3r::GUI::BitmapComboBox** ctrl,
|
|
wxWindow* parent,
|
|
const std::string& first_item/* = ""*/,
|
|
wxPoint pos/* = wxDefaultPosition*/,
|
|
wxSize size/* = wxDefaultSize*/,
|
|
bool use_thin_icon/* = false*/)
|
|
{
|
|
std::vector<wxBitmap*> icons = get_extruder_color_icons(use_thin_icon);
|
|
|
|
if (!*ctrl) {
|
|
*ctrl = new Slic3r::GUI::BitmapComboBox(parent, wxID_ANY, wxEmptyString, pos, size, 0, nullptr, wxCB_READONLY);
|
|
Slic3r::GUI::wxGetApp().UpdateDarkUI(*ctrl);
|
|
}
|
|
else
|
|
{
|
|
(*ctrl)->SetPosition(pos);
|
|
(*ctrl)->SetMinSize(size);
|
|
(*ctrl)->SetSize(size);
|
|
(*ctrl)->Clear();
|
|
}
|
|
if (first_item.empty())
|
|
(*ctrl)->Hide(); // to avoid unwanted rendering before layout (ExtruderSequenceDialog)
|
|
|
|
if (icons.empty() && !first_item.empty()) {
|
|
(*ctrl)->Append(_(first_item), wxNullBitmap);
|
|
return;
|
|
}
|
|
|
|
// For ObjectList we use short extruder name (just a number)
|
|
const bool use_full_item_name = dynamic_cast<Slic3r::GUI::ObjectList*>(parent) == nullptr;
|
|
|
|
int i = 0;
|
|
wxString str = _(L("Extruder"));
|
|
for (wxBitmap* bmp : icons) {
|
|
if (i == 0) {
|
|
if (!first_item.empty())
|
|
(*ctrl)->Append(_(first_item), *bmp);
|
|
++i;
|
|
}
|
|
|
|
(*ctrl)->Append(use_full_item_name
|
|
? Slic3r::GUI::from_u8((boost::format("%1% %2%") % str % i).str())
|
|
: wxString::Format("%d", i), *bmp);
|
|
++i;
|
|
}
|
|
(*ctrl)->SetSelection(0);
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// LockButton
|
|
// ----------------------------------------------------------------------------
|
|
|
|
LockButton::LockButton( wxWindow *parent,
|
|
wxWindowID id,
|
|
const wxPoint& pos /*= wxDefaultPosition*/,
|
|
const wxSize& size /*= wxDefaultSize*/):
|
|
wxButton(parent, id, wxEmptyString, pos, size, wxBU_EXACTFIT | wxNO_BORDER)
|
|
{
|
|
m_bmp_lock_closed = ScalableBitmap(this, "lock_normal");
|
|
m_bmp_lock_closed_f = ScalableBitmap(this, "lock_hover");
|
|
m_bmp_lock_open = ScalableBitmap(this, "unlock_normal");
|
|
m_bmp_lock_open_f = ScalableBitmap(this, "unlock_hover");
|
|
|
|
Slic3r::GUI::wxGetApp().UpdateDarkUI(this);
|
|
SetBitmap(m_bmp_lock_open.bmp());
|
|
SetBitmapDisabled(m_bmp_lock_open.bmp());
|
|
SetBitmapHover(m_bmp_lock_closed_f.bmp());
|
|
|
|
//button events
|
|
Bind(wxEVT_BUTTON, &LockButton::OnButton, this);
|
|
}
|
|
|
|
void LockButton::OnButton(wxCommandEvent& event)
|
|
{
|
|
if (m_disabled)
|
|
return;
|
|
|
|
m_is_pushed = !m_is_pushed;
|
|
update_button_bitmaps();
|
|
|
|
event.Skip();
|
|
}
|
|
|
|
void LockButton::SetLock(bool lock)
|
|
{
|
|
m_is_pushed = lock;
|
|
update_button_bitmaps();
|
|
}
|
|
|
|
void LockButton::msw_rescale()
|
|
{
|
|
m_bmp_lock_closed.msw_rescale();
|
|
m_bmp_lock_closed_f.msw_rescale();
|
|
m_bmp_lock_open.msw_rescale();
|
|
m_bmp_lock_open_f.msw_rescale();
|
|
|
|
update_button_bitmaps();
|
|
}
|
|
|
|
void LockButton::update_button_bitmaps()
|
|
{
|
|
Slic3r::GUI::wxGetApp().UpdateDarkUI(this);
|
|
SetBitmap(m_is_pushed ? m_bmp_lock_closed.bmp() : m_bmp_lock_open.bmp());
|
|
SetBitmapHover(m_is_pushed ? m_bmp_lock_closed_f.bmp() : m_bmp_lock_open_f.bmp());
|
|
|
|
Refresh();
|
|
Update();
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// ModeButton
|
|
// ----------------------------------------------------------------------------
|
|
|
|
ModeButton::ModeButton( wxWindow * parent,
|
|
wxWindowID id,
|
|
const std::string& icon_name /* = ""*/,
|
|
const wxString& mode /* = wxEmptyString*/,
|
|
const wxSize& size /* = wxDefaultSize*/,
|
|
const wxPoint& pos /* = wxDefaultPosition*/) :
|
|
ScalableButton(parent, id, icon_name, mode, size, pos, wxBU_EXACTFIT)
|
|
{
|
|
Init(mode);
|
|
}
|
|
|
|
ModeButton::ModeButton( wxWindow* parent,
|
|
const wxString& mode/* = wxEmptyString*/,
|
|
const std::string& icon_name/* = ""*/,
|
|
int px_cnt/* = 16*/) :
|
|
ScalableButton(parent, wxID_ANY, ScalableBitmap(parent, icon_name, px_cnt), mode, wxBU_EXACTFIT)
|
|
{
|
|
Init(mode);
|
|
}
|
|
|
|
void ModeButton::Init(const wxString &mode)
|
|
{
|
|
std::string mode_str = std::string(mode.ToUTF8());
|
|
//m_tt_focused = Slic3r::GUI::from_u8((boost::format(_utf8(L("Switch to the %s mode"))) % mode_str).str());
|
|
//m_tt_selected = Slic3r::GUI::from_u8((boost::format(_utf8(L("Current mode is %s"))) % mode_str).str());
|
|
|
|
SetBitmapMargins(3, 0);
|
|
|
|
//button events
|
|
Bind(wxEVT_BUTTON, &ModeButton::OnButton, this);
|
|
Bind(wxEVT_ENTER_WINDOW, &ModeButton::OnEnterBtn, this);
|
|
Bind(wxEVT_LEAVE_WINDOW, &ModeButton::OnLeaveBtn, this);
|
|
}
|
|
|
|
void ModeButton::OnButton(wxCommandEvent& event)
|
|
{
|
|
m_is_selected = true;
|
|
focus_button(m_is_selected);
|
|
|
|
event.Skip();
|
|
}
|
|
|
|
void ModeButton::SetState(const bool state)
|
|
{
|
|
m_is_selected = state;
|
|
focus_button(m_is_selected);
|
|
SetToolTip(state ? m_tt_selected : m_tt_focused);
|
|
}
|
|
|
|
void ModeButton::focus_button(const bool focus)
|
|
{
|
|
const wxFont& new_font = focus ?
|
|
Slic3r::GUI::wxGetApp().bold_font() :
|
|
Slic3r::GUI::wxGetApp().normal_font();
|
|
|
|
SetFont(new_font);
|
|
#ifdef _WIN32
|
|
GetParent()->Refresh(); // force redraw a background of the selected mode button
|
|
#else
|
|
SetForegroundColour(wxSystemSettings::GetColour(focus ? wxSYS_COLOUR_BTNTEXT :
|
|
#if defined (__linux__) && defined (__WXGTK3__)
|
|
wxSYS_COLOUR_GRAYTEXT
|
|
#elif defined (__linux__) && defined (__WXGTK2__)
|
|
wxSYS_COLOUR_BTNTEXT
|
|
#else
|
|
wxSYS_COLOUR_BTNSHADOW
|
|
#endif
|
|
));
|
|
#endif /* no _WIN32 */
|
|
|
|
Refresh();
|
|
Update();
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// ModeSizer
|
|
// ----------------------------------------------------------------------------
|
|
|
|
ModeSizer::ModeSizer(wxWindow *parent, int hgap/* = 0*/) :
|
|
wxFlexGridSizer(3, 0, hgap),
|
|
m_parent(parent),
|
|
m_hgap_unscaled((double)(hgap)/em_unit(parent))
|
|
{
|
|
SetFlexibleDirection(wxHORIZONTAL);
|
|
|
|
std::vector < std::pair < wxString, std::string >> buttons = {
|
|
//{_(L("Simple")), "mode_simple"},
|
|
//{_(L("Advanced")), "mode_advanced"},
|
|
//{_CTX(L_CONTEXT("Advanced", "Mode"), "Mode"), "mode_advanced"}
|
|
};
|
|
|
|
auto modebtnfn = [](wxCommandEvent &event, int mode_id) {
|
|
Slic3r::GUI::wxGetApp().save_mode(mode_id);
|
|
event.Skip();
|
|
};
|
|
|
|
m_mode_btns.reserve(3);
|
|
for (const auto& button : buttons) {
|
|
m_mode_btns.push_back(new ModeButton(parent, button.first, button.second, mode_icon_px_size()));
|
|
|
|
m_mode_btns.back()->Bind(wxEVT_BUTTON, std::bind(modebtnfn, std::placeholders::_1, int(m_mode_btns.size() - 1)));
|
|
Add(m_mode_btns.back());
|
|
}
|
|
}
|
|
|
|
void ModeSizer::SetMode(const int mode)
|
|
{
|
|
for (size_t m = 0; m < m_mode_btns.size(); m++)
|
|
m_mode_btns[m]->SetState(int(m) == mode);
|
|
}
|
|
|
|
void ModeSizer::set_items_flag(int flag)
|
|
{
|
|
for (wxSizerItem* item : this->GetChildren())
|
|
item->SetFlag(flag);
|
|
}
|
|
|
|
void ModeSizer::set_items_border(int border)
|
|
{
|
|
for (wxSizerItem* item : this->GetChildren())
|
|
item->SetBorder(border);
|
|
}
|
|
|
|
void ModeSizer::msw_rescale()
|
|
{
|
|
this->SetHGap(std::lround(m_hgap_unscaled * em_unit(m_parent)));
|
|
for (size_t m = 0; m < m_mode_btns.size(); m++)
|
|
m_mode_btns[m]->msw_rescale();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// MenuWithSeparators
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void MenuWithSeparators::DestroySeparators()
|
|
{
|
|
if (m_separator_frst) {
|
|
Destroy(m_separator_frst);
|
|
m_separator_frst = nullptr;
|
|
}
|
|
|
|
if (m_separator_scnd) {
|
|
Destroy(m_separator_scnd);
|
|
m_separator_scnd = nullptr;
|
|
}
|
|
}
|
|
|
|
void MenuWithSeparators::SetFirstSeparator()
|
|
{
|
|
m_separator_frst = this->AppendSeparator();
|
|
}
|
|
|
|
void MenuWithSeparators::SetSecondSeparator()
|
|
{
|
|
m_separator_scnd = this->AppendSeparator();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// BambuBitmap
|
|
// ----------------------------------------------------------------------------
|
|
ScalableBitmap::ScalableBitmap( wxWindow *parent,
|
|
const std::string& icon_name/* = ""*/,
|
|
const int px_cnt/* = 16*/,
|
|
const bool grayscale/* = false*/,
|
|
const bool resize/* = false*/,
|
|
const bool bitmap2/* = false*/,
|
|
const std::vector<std::string>& new_color/* = vector<std::string>*/) :
|
|
m_parent(parent), m_icon_name(icon_name),
|
|
m_px_cnt(px_cnt), m_grayscale(grayscale), m_resize(resize) // BBS: support resize by fill border
|
|
{
|
|
m_bmp = create_scaled_bitmap(icon_name, parent, px_cnt, m_grayscale, std::string(), false, resize, bitmap2, new_color);
|
|
if (px_cnt == 0) {
|
|
m_px_cnt = m_bmp.GetHeight(); // scale
|
|
unsigned int height = (unsigned int) (parent->FromDIP(m_px_cnt) + 0.5f);
|
|
if (height != GetBmpHeight())
|
|
msw_rescale();
|
|
}
|
|
}
|
|
|
|
wxSize ScalableBitmap::GetBmpSize() const
|
|
{
|
|
#ifdef __APPLE__
|
|
return m_bmp.GetScaledSize();
|
|
#else
|
|
return m_bmp.GetSize();
|
|
#endif
|
|
}
|
|
|
|
int ScalableBitmap::GetBmpWidth() const
|
|
{
|
|
#ifdef __APPLE__
|
|
return m_bmp.GetScaledWidth();
|
|
#else
|
|
return m_bmp.GetWidth();
|
|
#endif
|
|
}
|
|
|
|
int ScalableBitmap::GetBmpHeight() const
|
|
{
|
|
#ifdef __APPLE__
|
|
return m_bmp.GetScaledHeight();
|
|
#else
|
|
return m_bmp.GetHeight();
|
|
#endif
|
|
}
|
|
|
|
|
|
void ScalableBitmap::msw_rescale()
|
|
{
|
|
// BBS: support resize by fill border
|
|
m_bmp = create_scaled_bitmap(m_icon_name, m_parent, m_px_cnt, m_grayscale, std::string(), false, m_resize);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// BambuButton
|
|
// ----------------------------------------------------------------------------
|
|
|
|
ScalableButton::ScalableButton( wxWindow * parent,
|
|
wxWindowID id,
|
|
const std::string& icon_name /*= ""*/,
|
|
const wxString& label /* = wxEmptyString*/,
|
|
const wxSize& size /* = wxDefaultSize*/,
|
|
const wxPoint& pos /* = wxDefaultPosition*/,
|
|
long style /*= wxBU_EXACTFIT | wxNO_BORDER*/,
|
|
bool use_default_disabled_bitmap/* = false*/,
|
|
int bmp_px_cnt/* = 16*/) :
|
|
m_parent(parent),
|
|
m_current_icon_name(icon_name),
|
|
m_use_default_disabled_bitmap (use_default_disabled_bitmap),
|
|
m_px_cnt(bmp_px_cnt),
|
|
m_has_border(!(style & wxNO_BORDER))
|
|
{
|
|
SetBackgroundColour(StaticBox::GetParentBackgroundColor(parent));
|
|
Create(parent, id, label, pos, size, style);
|
|
Slic3r::GUI::wxGetApp().UpdateDarkUI(this);
|
|
|
|
if (!icon_name.empty()) {
|
|
SetBitmap(create_scaled_bitmap(icon_name, parent, m_px_cnt));
|
|
if (m_use_default_disabled_bitmap)
|
|
SetBitmapDisabled(create_scaled_bitmap(m_current_icon_name, m_parent, m_px_cnt, true));
|
|
if (!label.empty())
|
|
SetBitmapMargins(int(0.5* em_unit(parent)), 0);
|
|
}
|
|
|
|
if (size != wxDefaultSize)
|
|
{
|
|
const int em = em_unit(parent);
|
|
m_width = size.x/em;
|
|
m_height= size.y/em;
|
|
}
|
|
}
|
|
|
|
|
|
ScalableButton::ScalableButton( wxWindow * parent,
|
|
wxWindowID id,
|
|
const ScalableBitmap& bitmap,
|
|
const wxString& label /*= wxEmptyString*/,
|
|
long style /*= wxBU_EXACTFIT | wxNO_BORDER*/) :
|
|
m_parent(parent),
|
|
m_current_icon_name(bitmap.name()),
|
|
m_px_cnt(bitmap.px_cnt()),
|
|
m_has_border(!(style& wxNO_BORDER))
|
|
{
|
|
Create(parent, id, label, wxDefaultPosition, wxDefaultSize, style);
|
|
Slic3r::GUI::wxGetApp().UpdateDarkUI(this);
|
|
|
|
SetBitmap(bitmap.bmp());
|
|
}
|
|
|
|
void ScalableButton::SetBitmap_(const ScalableBitmap& bmp)
|
|
{
|
|
SetBitmap(bmp.bmp());
|
|
m_current_icon_name = bmp.name();
|
|
}
|
|
|
|
bool ScalableButton::SetBitmap_(const std::string& bmp_name)
|
|
{
|
|
m_current_icon_name = bmp_name;
|
|
if (m_current_icon_name.empty())
|
|
return false;
|
|
|
|
wxBitmap bmp = create_scaled_bitmap(m_current_icon_name, m_parent, m_px_cnt);
|
|
SetBitmap(bmp);
|
|
SetBitmapCurrent(bmp);
|
|
SetBitmapPressed(bmp);
|
|
SetBitmapFocus(bmp);
|
|
if (m_use_default_disabled_bitmap)
|
|
SetBitmapDisabled(create_scaled_bitmap(m_current_icon_name, m_parent, m_px_cnt, true));
|
|
return true;
|
|
}
|
|
|
|
void ScalableButton::SetBitmapDisabled_(const ScalableBitmap& bmp)
|
|
{
|
|
SetBitmapDisabled(bmp.bmp());
|
|
m_disabled_icon_name = bmp.name();
|
|
}
|
|
|
|
int ScalableButton::GetBitmapHeight()
|
|
{
|
|
#ifdef __APPLE__
|
|
return GetBitmap().GetScaledHeight();
|
|
#else
|
|
return GetBitmap().GetHeight();
|
|
#endif
|
|
}
|
|
|
|
void ScalableButton::UseDefaultBitmapDisabled()
|
|
{
|
|
m_use_default_disabled_bitmap = true;
|
|
SetBitmapDisabled(create_scaled_bitmap(m_current_icon_name, m_parent, m_px_cnt, true));
|
|
}
|
|
|
|
void ScalableButton::msw_rescale()
|
|
{
|
|
Slic3r::GUI::wxGetApp().UpdateDarkUI(this, m_has_border);
|
|
|
|
if (!m_current_icon_name.empty()) {
|
|
wxBitmap bmp = create_scaled_bitmap(m_current_icon_name, m_parent, m_px_cnt);
|
|
SetBitmap(bmp);
|
|
// BBS: why disappear on hover? why current HBITMAP differ from other
|
|
//SetBitmapCurrent(bmp);
|
|
//SetBitmapPressed(bmp);
|
|
//SetBitmapFocus(bmp);
|
|
if (!m_disabled_icon_name.empty())
|
|
SetBitmapDisabled(create_scaled_bitmap(m_disabled_icon_name, m_parent, m_px_cnt));
|
|
else if (m_use_default_disabled_bitmap)
|
|
SetBitmapDisabled(create_scaled_bitmap(m_current_icon_name, m_parent, m_px_cnt, true));
|
|
}
|
|
|
|
if (m_width > 0 || m_height>0)
|
|
{
|
|
const int em = em_unit(m_parent);
|
|
wxSize size(m_width * em, m_height * em);
|
|
SetMinSize(size);
|
|
}
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// BlinkingBitmap
|
|
// ----------------------------------------------------------------------------
|
|
|
|
BlinkingBitmap::BlinkingBitmap(wxWindow* parent, const std::string& icon_name) :
|
|
wxStaticBitmap(parent, wxID_ANY, wxNullBitmap, wxDefaultPosition, wxSize(int(1.6 * Slic3r::GUI::wxGetApp().em_unit()), -1))
|
|
{
|
|
bmp = ScalableBitmap(parent, icon_name);
|
|
}
|
|
|
|
void BlinkingBitmap::msw_rescale()
|
|
{
|
|
bmp.msw_rescale();
|
|
this->SetSize(bmp.GetBmpSize());
|
|
this->SetMinSize(bmp.GetBmpSize());
|
|
}
|
|
|
|
void BlinkingBitmap::invalidate()
|
|
{
|
|
this->SetBitmap(wxNullBitmap);
|
|
}
|
|
|
|
void BlinkingBitmap::activate()
|
|
{
|
|
this->SetBitmap(bmp.bmp());
|
|
show = true;
|
|
}
|
|
|
|
void BlinkingBitmap::blink()
|
|
{
|
|
show = !show;
|
|
this->SetBitmap(show ? bmp.bmp() : wxNullBitmap);
|
|
}
|
|
|
|
|
|
wxIMPLEMENT_CLASS(ImageTransientPopup,PopupWindow);
|
|
|
|
wxBEGIN_EVENT_TABLE(ImageTransientPopup,PopupWindow)
|
|
EVT_MOUSE_EVENTS( ImageTransientPopup::OnMouse )
|
|
EVT_SIZE( ImageTransientPopup::OnSize )
|
|
EVT_SET_FOCUS( ImageTransientPopup::OnSetFocus )
|
|
EVT_KILL_FOCUS( ImageTransientPopup::OnKillFocus )
|
|
wxEND_EVENT_TABLE()
|
|
|
|
ImageTransientPopup::ImageTransientPopup( wxWindow *parent, bool scrolled, wxBitmap bmp)
|
|
:PopupWindow( parent,
|
|
wxBORDER_NONE |
|
|
wxPU_CONTAINS_CONTROLS )
|
|
{
|
|
m_panel = new wxScrolledWindow( this, wxID_ANY );
|
|
m_panel->SetBackgroundColour( *wxLIGHT_GREY );
|
|
|
|
// Keep this code to verify if mouse events work, they're required if
|
|
// you're making a control like a combobox where the items are highlighted
|
|
// under the cursor, the m_panel is set focus in the Popup() function
|
|
m_panel->Bind(wxEVT_MOTION, &ImageTransientPopup::OnMouse, this);
|
|
|
|
m_image = new wxStaticBitmap(m_panel,
|
|
wxID_ANY, bmp);
|
|
|
|
wxBoxSizer *topSizer = new wxBoxSizer( wxVERTICAL );
|
|
topSizer->Add(m_image, 1, wxCENTRE | wxALL | wxEXPAND, 0);
|
|
topSizer->SetMinSize(300, 300);
|
|
|
|
if ( scrolled )
|
|
{
|
|
// Add a big window to ensure that scrollbars are shown when we set the
|
|
// panel size to a lesser size below.
|
|
topSizer->Add(new wxPanel(m_panel, wxID_ANY, wxDefaultPosition,
|
|
wxSize(600, 600)));
|
|
}
|
|
|
|
m_panel->SetSizer( topSizer );
|
|
if ( scrolled )
|
|
{
|
|
// Set the fixed size to ensure that the scrollbars are shown.
|
|
m_panel->SetSize(300, 300);
|
|
|
|
// And also actually enable them.
|
|
m_panel->SetScrollRate(10, 10);
|
|
}
|
|
else
|
|
{
|
|
// Use the fitting size for the panel if we don't need scrollbars.
|
|
topSizer->Fit(m_panel);
|
|
}
|
|
|
|
SetClientSize(m_panel->GetSize());
|
|
}
|
|
|
|
ImageTransientPopup::~ImageTransientPopup()
|
|
{
|
|
}
|
|
|
|
void ImageTransientPopup::SetImage(wxBitmap bmp)
|
|
{
|
|
m_image->SetBitmap(bmp);
|
|
m_panel->Layout();
|
|
}
|
|
|
|
void ImageTransientPopup::Popup(wxWindow* WXUNUSED(focus))
|
|
{
|
|
PopupWindow::Popup();
|
|
}
|
|
|
|
void ImageTransientPopup::OnDismiss()
|
|
{
|
|
PopupWindow::OnDismiss();
|
|
}
|
|
|
|
bool ImageTransientPopup::ProcessLeftDown(wxMouseEvent& event)
|
|
{
|
|
return PopupWindow::ProcessLeftDown(event);
|
|
}
|
|
bool ImageTransientPopup::Show( bool show )
|
|
{
|
|
return PopupWindow::Show(show);
|
|
}
|
|
|
|
void ImageTransientPopup::OnSize(wxSizeEvent &event)
|
|
{
|
|
event.Skip();
|
|
}
|
|
|
|
void ImageTransientPopup::OnSetFocus(wxFocusEvent &event)
|
|
{
|
|
event.Skip();
|
|
}
|
|
|
|
void ImageTransientPopup::OnKillFocus(wxFocusEvent &event)
|
|
{
|
|
event.Skip();
|
|
}
|
|
|
|
void ImageTransientPopup::OnMouse(wxMouseEvent &event)
|
|
{
|
|
event.Skip();
|
|
}
|
|
|
|
|
|
|
|
|