2019-09-27 12:52:19 +00:00
|
|
|
#include "libslic3r/libslic3r.h"
|
|
|
|
#include "Mouse3DController.hpp"
|
|
|
|
|
|
|
|
#if ENABLE_3DCONNEXION_DEVICES
|
|
|
|
|
2019-10-03 09:38:31 +00:00
|
|
|
#include "Camera.hpp"
|
2019-09-27 12:52:19 +00:00
|
|
|
#include "GUI_App.hpp"
|
|
|
|
#include "PresetBundle.hpp"
|
2019-10-03 08:26:28 +00:00
|
|
|
#include "AppConfig.hpp"
|
2019-09-27 12:52:19 +00:00
|
|
|
|
|
|
|
#include <wx/glcanvas.h>
|
|
|
|
|
2019-10-02 13:55:26 +00:00
|
|
|
#include <boost/nowide/convert.hpp>
|
2019-10-04 08:59:27 +00:00
|
|
|
#include <boost/log/trivial.hpp>
|
2019-10-02 13:55:26 +00:00
|
|
|
#include "I18N.hpp"
|
|
|
|
|
2019-10-09 12:01:13 +00:00
|
|
|
#include <bitset>
|
|
|
|
|
2019-09-30 12:58:51 +00:00
|
|
|
// WARN: If updating these lists, please also update resources/udev/90-3dconnexion.rules
|
|
|
|
|
2019-09-27 12:52:19 +00:00
|
|
|
static const std::vector<int> _3DCONNEXION_VENDORS =
|
|
|
|
{
|
|
|
|
0x046d, // LOGITECH = 1133 // Logitech (3Dconnexion is made by Logitech)
|
|
|
|
0x256F // 3DCONNECTION = 9583 // 3Dconnexion
|
|
|
|
};
|
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
// See: https://github.com/FreeSpacenav/spacenavd/blob/a9eccf34e7cac969ee399f625aef827f4f4aaec6/src/dev.c#L202
|
2019-09-27 12:52:19 +00:00
|
|
|
static const std::vector<int> _3DCONNEXION_DEVICES =
|
|
|
|
{
|
2019-10-08 11:38:08 +00:00
|
|
|
0xc603, /* 50691 spacemouse plus XT */
|
|
|
|
0xc605, /* 50693 cadman */
|
|
|
|
0xc606, /* 50694 spacemouse classic */
|
|
|
|
0xc621, /* 50721 spaceball 5000 */
|
|
|
|
0xc623, /* 50723 space traveller */
|
|
|
|
0xc625, /* 50725 space pilot */
|
|
|
|
0xc626, /* 50726 space navigator *TESTED* */
|
|
|
|
0xc627, /* 50727 space explorer */
|
|
|
|
0xc628, /* 50728 space navigator for notebooks*/
|
|
|
|
0xc629, /* 50729 space pilot pro*/
|
|
|
|
0xc62b, /* 50731 space mouse pro*/
|
2019-10-09 12:01:13 +00:00
|
|
|
0xc62e, /* 50734 spacemouse wireless (USB cable) *TESTED* */
|
|
|
|
0xc62f, /* 50735 spacemouse wireless receiver */
|
2019-10-08 11:38:08 +00:00
|
|
|
0xc631, /* 50737 spacemouse pro wireless *TESTED* */
|
|
|
|
0xc632, /* 50738 spacemouse pro wireless receiver */
|
|
|
|
0xc633, /* 50739 spacemouse enterprise */
|
2019-10-09 12:01:13 +00:00
|
|
|
0xc635, /* 50741 spacemouse compact *TESTED* */
|
2019-10-08 11:38:08 +00:00
|
|
|
0xc636, /* 50742 spacemouse module */
|
|
|
|
0xc640, /* 50752 nulooq */
|
|
|
|
|
2019-10-09 12:01:13 +00:00
|
|
|
// 0xc652, /* 50770 3Dconnexion universal receiver */
|
2019-09-27 12:52:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
|
|
|
const double Mouse3DController::State::DefaultTranslationScale = 2.5;
|
2019-10-08 12:32:05 +00:00
|
|
|
const double Mouse3DController::State::MaxTranslationDeadzone = 0.1;
|
|
|
|
const double Mouse3DController::State::DefaultTranslationDeadzone = 0.5 * Mouse3DController::State::MaxTranslationDeadzone;
|
2019-10-08 07:52:56 +00:00
|
|
|
const float Mouse3DController::State::DefaultRotationScale = 1.0f;
|
2019-10-08 12:32:05 +00:00
|
|
|
const float Mouse3DController::State::MaxRotationDeadzone = 0.1f;
|
|
|
|
const float Mouse3DController::State::DefaultRotationDeadzone = 0.5f * Mouse3DController::State::MaxRotationDeadzone;
|
2019-09-27 12:52:19 +00:00
|
|
|
|
|
|
|
Mouse3DController::State::State()
|
2019-10-08 12:32:05 +00:00
|
|
|
: m_translation_params(DefaultTranslationScale, DefaultTranslationDeadzone)
|
|
|
|
, m_rotation_params(DefaultRotationScale, DefaultRotationDeadzone)
|
2019-10-08 06:44:50 +00:00
|
|
|
, m_mouse_wheel_counter(0)
|
2019-10-09 13:23:30 +00:00
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
|
|
#if ENABLE_3DCONNEXION_DEVICES_DEBUG_OUTPUT
|
|
|
|
, m_translation_queue_max_size(0)
|
|
|
|
, m_rotation_queue_max_size(0)
|
|
|
|
, m_buttons_queue_max_size(0)
|
|
|
|
#endif // ENABLE_3DCONNEXION_DEVICES_DEBUG_OUTPUT
|
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
2019-10-08 06:44:50 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Mouse3DController::State::process_mouse_wheel()
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
2019-10-08 11:38:08 +00:00
|
|
|
if (m_mouse_wheel_counter == 0)
|
|
|
|
return false;
|
|
|
|
else if (!m_rotation.empty())
|
2019-10-08 06:44:50 +00:00
|
|
|
{
|
|
|
|
--m_mouse_wheel_counter;
|
|
|
|
return true;
|
|
|
|
}
|
2019-10-08 11:38:08 +00:00
|
|
|
else
|
2019-10-08 06:44:50 +00:00
|
|
|
{
|
|
|
|
m_mouse_wheel_counter = 0;
|
|
|
|
return true;
|
|
|
|
}
|
2019-09-27 12:52:19 +00:00
|
|
|
}
|
|
|
|
|
2019-10-03 09:38:31 +00:00
|
|
|
bool Mouse3DController::State::apply(Camera& camera)
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
|
|
|
if (!wxGetApp().IsActive())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
if (has_translation())
|
|
|
|
{
|
2019-10-08 06:44:50 +00:00
|
|
|
const Vec3d& translation = m_translation.front();
|
2019-10-08 12:32:05 +00:00
|
|
|
camera.set_target(camera.get_target() + m_translation_params.scale * (translation(0) * camera.get_dir_right() + translation(1) * camera.get_dir_forward() + translation(2) * camera.get_dir_up()));
|
2019-10-08 06:44:50 +00:00
|
|
|
m_translation.pop();
|
2019-09-27 12:52:19 +00:00
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_rotation())
|
|
|
|
{
|
2019-10-08 06:44:50 +00:00
|
|
|
const Vec3f& rotation = m_rotation.front();
|
2019-10-08 12:32:05 +00:00
|
|
|
float theta = m_rotation_params.scale * rotation(0);
|
|
|
|
float phi = m_rotation_params.scale * rotation(2);
|
2019-09-27 12:52:19 +00:00
|
|
|
float sign = camera.inverted_phi ? -1.0f : 1.0f;
|
|
|
|
camera.phi += sign * phi;
|
|
|
|
camera.set_theta(camera.get_theta() + theta, wxGetApp().preset_bundle->printers.get_edited_preset().printer_technology() != ptSLA);
|
2019-10-08 06:44:50 +00:00
|
|
|
m_rotation.pop();
|
|
|
|
|
2019-09-27 12:52:19 +00:00
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
|
2019-10-09 12:18:43 +00:00
|
|
|
if (has_button())
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
2019-10-08 06:44:50 +00:00
|
|
|
unsigned int button = m_buttons.front();
|
|
|
|
switch (button)
|
2019-09-30 13:58:45 +00:00
|
|
|
{
|
2019-10-08 06:44:50 +00:00
|
|
|
case 0: { camera.update_zoom(1.0); break; }
|
|
|
|
case 1: { camera.update_zoom(-1.0); break; }
|
|
|
|
default: { break; }
|
2019-09-30 13:58:45 +00:00
|
|
|
}
|
2019-10-08 06:44:50 +00:00
|
|
|
m_buttons.pop();
|
2019-09-27 12:52:19 +00:00
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
Mouse3DController::Mouse3DController()
|
|
|
|
: m_initialized(false)
|
|
|
|
, m_device(nullptr)
|
2019-10-03 08:26:28 +00:00
|
|
|
, m_device_str("")
|
2019-09-27 12:52:19 +00:00
|
|
|
, m_running(false)
|
2019-10-02 13:55:26 +00:00
|
|
|
, m_settings_dialog(false)
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void Mouse3DController::init()
|
|
|
|
{
|
|
|
|
if (m_initialized)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Initialize the hidapi library
|
|
|
|
int res = hid_init();
|
|
|
|
if (res != 0)
|
2019-10-04 08:59:27 +00:00
|
|
|
{
|
|
|
|
BOOST_LOG_TRIVIAL(error) << "Unable to initialize hidapi library";
|
2019-09-27 12:52:19 +00:00
|
|
|
return;
|
2019-10-04 08:59:27 +00:00
|
|
|
}
|
2019-09-27 12:52:19 +00:00
|
|
|
|
|
|
|
m_initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Mouse3DController::shutdown()
|
|
|
|
{
|
|
|
|
if (!m_initialized)
|
|
|
|
return;
|
|
|
|
|
|
|
|
stop();
|
|
|
|
disconnect_device();
|
|
|
|
|
|
|
|
// Finalize the hidapi library
|
|
|
|
hid_exit();
|
|
|
|
m_initialized = false;
|
|
|
|
}
|
|
|
|
|
2019-10-03 10:16:59 +00:00
|
|
|
bool Mouse3DController::apply(Camera& camera)
|
|
|
|
{
|
2019-10-04 08:59:27 +00:00
|
|
|
if (!m_initialized)
|
|
|
|
return false;
|
|
|
|
|
2019-10-03 10:16:59 +00:00
|
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
2019-10-04 08:59:27 +00:00
|
|
|
|
|
|
|
// check if the user unplugged the device
|
|
|
|
if (!m_running && is_device_connected())
|
|
|
|
{
|
|
|
|
disconnect_device();
|
|
|
|
// hides the settings dialog if the user re-plug the device
|
|
|
|
m_settings_dialog = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if the user plugged the device
|
|
|
|
if (connect_device())
|
|
|
|
start();
|
|
|
|
|
|
|
|
return is_device_connected() ? m_state.apply(camera) : false;
|
2019-10-03 10:16:59 +00:00
|
|
|
}
|
|
|
|
|
2019-10-03 09:38:31 +00:00
|
|
|
void Mouse3DController::render_settings_dialog(unsigned int canvas_width, unsigned int canvas_height) const
|
2019-10-02 13:55:26 +00:00
|
|
|
{
|
2019-10-03 09:38:31 +00:00
|
|
|
if (!m_running || !m_settings_dialog)
|
2019-10-02 13:55:26 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
ImGuiWrapper& imgui = *wxGetApp().imgui();
|
|
|
|
|
2019-10-03 09:38:31 +00:00
|
|
|
imgui.set_next_window_pos(0.5f * (float)canvas_width, 0.5f * (float)canvas_height, ImGuiCond_Always, 0.5f, 0.5f);
|
2019-10-02 13:55:26 +00:00
|
|
|
imgui.set_next_window_bg_alpha(0.5f);
|
|
|
|
|
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f);
|
|
|
|
|
|
|
|
imgui.begin(_(L("3Dconnexion settings")), ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoCollapse);
|
|
|
|
|
|
|
|
const ImVec4& color = ImGui::GetStyleColorVec4(ImGuiCol_Separator);
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_Text, color);
|
|
|
|
imgui.text(_(L("Device:")));
|
|
|
|
ImGui::PopStyleColor();
|
|
|
|
ImGui::SameLine();
|
2019-10-03 08:26:28 +00:00
|
|
|
imgui.text(m_device_str);
|
2019-10-02 13:55:26 +00:00
|
|
|
|
2019-10-08 07:52:56 +00:00
|
|
|
ImGui::Separator();
|
2019-10-02 13:55:26 +00:00
|
|
|
ImGui::PushStyleColor(ImGuiCol_Text, color);
|
|
|
|
imgui.text(_(L("Speed:")));
|
|
|
|
ImGui::PopStyleColor();
|
|
|
|
|
2019-10-08 12:32:05 +00:00
|
|
|
float translation_scale = (float)m_state.get_translation_scale() / State::DefaultTranslationScale;
|
|
|
|
if (ImGui::SliderFloat(_(L("Translation##1")), &translation_scale, 0.5f, 2.0f, "%.1f"))
|
|
|
|
m_state.set_translation_scale(State::DefaultTranslationScale * (double)translation_scale);
|
2019-10-02 13:55:26 +00:00
|
|
|
|
2019-10-08 12:32:05 +00:00
|
|
|
float rotation_scale = m_state.get_rotation_scale() / State::DefaultRotationScale;
|
|
|
|
if (ImGui::SliderFloat(_(L("Rotation##1")), &rotation_scale, 0.5f, 2.0f, "%.1f"))
|
|
|
|
m_state.set_rotation_scale(State::DefaultRotationScale * rotation_scale);
|
|
|
|
|
|
|
|
ImGui::Separator();
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_Text, color);
|
|
|
|
imgui.text(_(L("Deadzone:")));
|
|
|
|
ImGui::PopStyleColor();
|
|
|
|
|
|
|
|
float translation_deadzone = (float)m_state.get_translation_deadzone();
|
|
|
|
if (ImGui::SliderFloat(_(L("Translation##2")), &translation_deadzone, 0.0f, (float)State::MaxTranslationDeadzone, "%.2f"))
|
|
|
|
m_state.set_translation_deadzone((double)translation_deadzone);
|
|
|
|
|
|
|
|
float rotation_deadzone = m_state.get_rotation_deadzone();
|
|
|
|
if (ImGui::SliderFloat(_(L("Rotation##2")), &rotation_deadzone, 0.0f, State::MaxRotationDeadzone, "%.2f"))
|
|
|
|
m_state.set_rotation_deadzone(rotation_deadzone);
|
2019-10-02 13:55:26 +00:00
|
|
|
|
2019-10-09 12:18:43 +00:00
|
|
|
#if ENABLE_3DCONNEXION_DEVICES_DEBUG_OUTPUT
|
2019-10-09 13:23:30 +00:00
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
|
|
ImGui::Separator();
|
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
2019-10-09 12:18:43 +00:00
|
|
|
ImGui::Separator();
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_Text, color);
|
2019-10-09 13:23:30 +00:00
|
|
|
imgui.text("DEBUG:");
|
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
|
|
imgui.text("Vectors:");
|
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
2019-10-09 12:18:43 +00:00
|
|
|
ImGui::PopStyleColor();
|
|
|
|
Vec3f translation = m_state.get_translation().cast<float>();
|
|
|
|
Vec3f rotation = m_state.get_rotation();
|
|
|
|
unsigned int button = m_state.get_button();
|
|
|
|
ImGui::InputFloat3("Translation##3", translation.data(), "%.3f", ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputFloat3("Rotation##3", rotation.data(), "%.3f", ImGuiInputTextFlags_ReadOnly);
|
2019-10-09 13:23:30 +00:00
|
|
|
|
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_Text, color);
|
|
|
|
imgui.text("Queue size:");
|
|
|
|
ImGui::PopStyleColor();
|
|
|
|
|
|
|
|
int translation_size[2] = { (int)m_state.get_translation_queue_size(), (int)m_state.get_translation_queue_max_size() };
|
|
|
|
int rotation_size[2] = { (int)m_state.get_rotation_queue_size(), (int)m_state.get_rotation_queue_max_size() };
|
|
|
|
int buttons_size[2] = { (int)m_state.get_buttons_queue_size(), (int)m_state.get_buttons_queue_max_size() };
|
|
|
|
|
|
|
|
ImGui::InputInt2("Translation##4", translation_size, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt2("Rotation##4", rotation_size, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt2("Buttons", buttons_size, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
|
|
|
|
/*
|
|
|
|
int translation_size = (int)m_state.get_translation_queue_size();
|
|
|
|
int translation_max_size = (int)m_state.get_translation_queue_max_size();
|
|
|
|
int rotation_size = (int)m_state.get_rotation_queue_size();
|
|
|
|
int rotation_max_size = (int)m_state.get_rotation_queue_max_size();
|
|
|
|
int buttons_size = (int)m_state.get_buttons_queue_size();
|
|
|
|
int buttons_max_size = (int)m_state.get_buttons_queue_max_size();
|
|
|
|
|
|
|
|
ImGui::InputInt("Translation size", &translation_size, 0, 0, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt("Translation max", &translation_max_size, 0, 0, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt("Rotation size", &rotation_size, 0, 0, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt("Rotation max", &rotation_max_size, 0, 0, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt("Button size", &buttons_size, 0, 0, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::InputInt("Button max", &buttons_max_size, 0, 0, ImGuiInputTextFlags_ReadOnly);
|
|
|
|
*/
|
|
|
|
//@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
2019-10-09 12:18:43 +00:00
|
|
|
#endif // ENABLE_3DCONNEXION_DEVICES_DEBUG_OUTPUT
|
|
|
|
|
2019-10-02 13:55:26 +00:00
|
|
|
imgui.end();
|
|
|
|
|
|
|
|
ImGui::PopStyleVar();
|
|
|
|
}
|
|
|
|
|
2019-10-04 08:59:27 +00:00
|
|
|
bool Mouse3DController::connect_device()
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
2019-10-04 08:59:27 +00:00
|
|
|
if (is_device_connected())
|
|
|
|
return false;
|
2019-09-27 12:52:19 +00:00
|
|
|
|
|
|
|
// Enumerates devices
|
|
|
|
hid_device_info* devices = hid_enumerate(0, 0);
|
|
|
|
if (devices == nullptr)
|
2019-10-04 08:59:27 +00:00
|
|
|
{
|
|
|
|
BOOST_LOG_TRIVIAL(error) << "Unable to enumerate HID devices";
|
|
|
|
return false;
|
|
|
|
}
|
2019-09-27 12:52:19 +00:00
|
|
|
|
|
|
|
// Searches for 1st connected 3Dconnexion device
|
|
|
|
unsigned short vendor_id = 0;
|
|
|
|
unsigned short product_id = 0;
|
|
|
|
|
2019-10-09 12:18:43 +00:00
|
|
|
#if ENABLE_3DCONNEXION_DEVICES_DEBUG_OUTPUT
|
|
|
|
hid_device_info* cur = devices;
|
|
|
|
while (cur != nullptr)
|
|
|
|
{
|
|
|
|
std::cout << "Detected device '";
|
|
|
|
std::wcout << ((cur->manufacturer_string != nullptr) ? cur->manufacturer_string : L"Unknown");
|
|
|
|
std::cout << "::";
|
|
|
|
std::wcout << ((cur->product_string != nullptr) ? cur->product_string : L"Unknown");
|
|
|
|
std::cout << "' code: " << cur->vendor_id << "/" << cur->product_id << " (" << std::hex << cur->vendor_id << "/" << cur->product_id << std::dec << ")" << std::endl;
|
|
|
|
|
|
|
|
cur = cur->next;
|
|
|
|
}
|
|
|
|
#endif // ENABLE_3DCONNEXION_DEVICES_DEBUG_OUTPUT
|
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
hid_device_info* current = devices;
|
|
|
|
while (current != nullptr)
|
|
|
|
{
|
2019-09-27 12:52:19 +00:00
|
|
|
for (size_t i = 0; i < _3DCONNEXION_VENDORS.size(); ++i)
|
|
|
|
{
|
|
|
|
if (_3DCONNEXION_VENDORS[i] == current->vendor_id)
|
|
|
|
{
|
|
|
|
vendor_id = current->vendor_id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vendor_id != 0)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < _3DCONNEXION_DEVICES.size(); ++i)
|
|
|
|
{
|
|
|
|
if (_3DCONNEXION_DEVICES[i] == current->product_id)
|
|
|
|
{
|
|
|
|
product_id = current->product_id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (product_id == 0)
|
|
|
|
vendor_id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vendor_id != 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
current = current->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Free enumerated devices
|
|
|
|
hid_free_enumeration(devices);
|
|
|
|
|
|
|
|
if (vendor_id == 0)
|
2019-10-04 08:59:27 +00:00
|
|
|
return false;
|
2019-09-27 12:52:19 +00:00
|
|
|
|
|
|
|
// Open the 3Dconnexion device using the VID, PID
|
|
|
|
m_device = hid_open(vendor_id, product_id, nullptr);
|
2019-10-03 08:26:28 +00:00
|
|
|
|
|
|
|
if (m_device != nullptr)
|
|
|
|
{
|
|
|
|
std::vector<wchar_t> product(1024, 0);
|
|
|
|
hid_get_product_string(m_device, product.data(), 1024);
|
|
|
|
m_device_str = boost::nowide::narrow(product.data());
|
|
|
|
|
2019-10-04 08:59:27 +00:00
|
|
|
BOOST_LOG_TRIVIAL(info) << "Connected device: " << m_device_str;
|
|
|
|
|
|
|
|
// get device parameters from the config, if present
|
2019-10-08 12:32:05 +00:00
|
|
|
double translation_speed = 1.0;
|
|
|
|
float rotation_speed = 1.0;
|
|
|
|
double translation_deadzone = State::DefaultTranslationDeadzone;
|
|
|
|
float rotation_deadzone = State::DefaultRotationDeadzone;
|
|
|
|
wxGetApp().app_config->get_mouse_device_translation_speed(m_device_str, translation_speed);
|
|
|
|
wxGetApp().app_config->get_mouse_device_translation_deadzone(m_device_str, translation_deadzone);
|
|
|
|
wxGetApp().app_config->get_mouse_device_rotation_speed(m_device_str, rotation_speed);
|
|
|
|
wxGetApp().app_config->get_mouse_device_rotation_deadzone(m_device_str, rotation_deadzone);
|
2019-10-03 08:26:28 +00:00
|
|
|
// clamp to valid values
|
2019-10-08 12:32:05 +00:00
|
|
|
m_state.set_translation_scale(State::DefaultTranslationScale* std::max(0.5, std::min(2.0, translation_speed)));
|
|
|
|
m_state.set_translation_deadzone(std::max(0.0, std::min(State::MaxTranslationDeadzone, translation_deadzone)));
|
|
|
|
m_state.set_rotation_scale(State::DefaultRotationScale* std::max(0.5f, std::min(2.0f, rotation_speed)));
|
|
|
|
m_state.set_rotation_deadzone(std::max(0.0f, std::min(State::MaxRotationDeadzone, rotation_deadzone)));
|
2019-10-03 08:26:28 +00:00
|
|
|
}
|
2019-10-04 08:59:27 +00:00
|
|
|
|
|
|
|
return (m_device != nullptr);
|
2019-09-27 12:52:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Mouse3DController::disconnect_device()
|
|
|
|
{
|
2019-10-04 08:59:27 +00:00
|
|
|
if (!is_device_connected())
|
2019-09-27 12:52:19 +00:00
|
|
|
return;
|
|
|
|
|
2019-10-04 08:59:27 +00:00
|
|
|
// Stop the secondary thread, if running
|
|
|
|
if (m_thread.joinable())
|
|
|
|
m_thread.join();
|
|
|
|
|
|
|
|
// Store current device parameters into the config
|
2019-10-08 12:32:05 +00:00
|
|
|
wxGetApp().app_config->set_mouse_device(m_device_str, m_state.get_translation_scale() / State::DefaultTranslationScale, m_state.get_translation_deadzone(),
|
|
|
|
m_state.get_rotation_scale() / State::DefaultRotationScale, m_state.get_rotation_deadzone());
|
2019-10-03 08:26:28 +00:00
|
|
|
wxGetApp().app_config->save();
|
|
|
|
|
2019-09-27 12:52:19 +00:00
|
|
|
// Close the 3Dconnexion device
|
|
|
|
hid_close(m_device);
|
|
|
|
m_device = nullptr;
|
2019-10-04 08:59:27 +00:00
|
|
|
|
|
|
|
BOOST_LOG_TRIVIAL(info) << "Disconnected device: " << m_device_str;
|
|
|
|
|
2019-10-03 08:26:28 +00:00
|
|
|
m_device_str = "";
|
2019-09-27 12:52:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Mouse3DController::start()
|
|
|
|
{
|
2019-10-04 08:59:27 +00:00
|
|
|
if (!is_device_connected() || m_running)
|
2019-09-27 12:52:19 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
m_thread = std::thread(&Mouse3DController::run, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Mouse3DController::run()
|
|
|
|
{
|
|
|
|
m_running = true;
|
|
|
|
while (m_running)
|
|
|
|
{
|
|
|
|
collect_input();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-04 05:58:01 +00:00
|
|
|
void Mouse3DController::collect_input()
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
2019-10-08 11:38:08 +00:00
|
|
|
DataPacket packet = { 0 };
|
|
|
|
int res = hid_read_timeout(m_device, packet.data(), packet.size(), 100);
|
2019-09-27 12:52:19 +00:00
|
|
|
if (res < 0)
|
|
|
|
{
|
|
|
|
// An error occourred (device detached from pc ?)
|
|
|
|
stop();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
if (!wxGetApp().IsActive())
|
|
|
|
return;
|
|
|
|
|
2019-10-03 10:16:59 +00:00
|
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
bool updated = false;
|
|
|
|
|
|
|
|
if (res == 7)
|
|
|
|
updated = handle_packet(packet);
|
|
|
|
else if (res == 13)
|
|
|
|
updated = handle_wireless_packet(packet);
|
|
|
|
else if (res > 0)
|
|
|
|
std::cout << "Got unknown data packet of length: " << res << ", code:" << (int)packet[0] << std::endl;
|
|
|
|
|
|
|
|
if (updated)
|
|
|
|
// ask for an idle event to update 3D scene
|
|
|
|
wxWakeUpIdle();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Mouse3DController::handle_packet(const DataPacket& packet)
|
|
|
|
{
|
|
|
|
switch (packet[0])
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
2019-10-08 11:38:08 +00:00
|
|
|
case 1: // Translation
|
|
|
|
{
|
|
|
|
if (handle_packet_translation(packet))
|
|
|
|
return true;
|
2019-10-07 07:31:23 +00:00
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2: // Rotation
|
2019-09-27 12:52:19 +00:00
|
|
|
{
|
2019-10-08 11:38:08 +00:00
|
|
|
if (handle_packet_rotation(packet, 1))
|
|
|
|
return true;
|
2019-09-27 12:52:19 +00:00
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 3: // Button
|
|
|
|
{
|
|
|
|
if (handle_packet_button(packet, 6))
|
|
|
|
return true;
|
2019-09-27 12:52:19 +00:00
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
std::cout << "Got unknown data packet of code: " << (int)packet[0] << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-09-27 12:52:19 +00:00
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Mouse3DController::handle_wireless_packet(const DataPacket& packet)
|
|
|
|
{
|
|
|
|
switch (packet[0])
|
|
|
|
{
|
|
|
|
case 1: // Translation + Rotation
|
|
|
|
{
|
|
|
|
bool updated = handle_packet_translation(packet);
|
|
|
|
updated |= handle_packet_rotation(packet, 7);
|
|
|
|
|
|
|
|
if (updated)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 3: // Button
|
|
|
|
{
|
|
|
|
if (handle_packet_button(packet, 12))
|
|
|
|
return true;
|
2019-10-03 08:26:28 +00:00
|
|
|
|
2019-09-27 12:52:19 +00:00
|
|
|
break;
|
|
|
|
}
|
2019-10-08 11:38:08 +00:00
|
|
|
default:
|
|
|
|
{
|
|
|
|
std::cout << "Got unknown data packet of code: " << (int)packet[0] << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-10-09 12:39:28 +00:00
|
|
|
double convert_input(unsigned char first, unsigned char second, double deadzone)
|
2019-10-08 11:38:08 +00:00
|
|
|
{
|
2019-10-09 12:01:13 +00:00
|
|
|
short value = first | second << 8;
|
2019-10-09 12:39:28 +00:00
|
|
|
double ret = (double)value / 350.0;
|
|
|
|
return (std::abs(ret) > deadzone) ? ret : 0.0;
|
2019-10-09 12:01:13 +00:00
|
|
|
}
|
2019-10-08 11:38:08 +00:00
|
|
|
|
|
|
|
bool Mouse3DController::handle_packet_translation(const DataPacket& packet)
|
|
|
|
{
|
2019-10-08 12:32:05 +00:00
|
|
|
double deadzone = m_state.get_translation_deadzone();
|
2019-10-09 12:39:28 +00:00
|
|
|
Vec3d translation(-convert_input(packet[1], packet[2], deadzone),
|
|
|
|
convert_input(packet[3], packet[4], deadzone),
|
|
|
|
convert_input(packet[5], packet[6], deadzone));
|
|
|
|
|
|
|
|
if (!translation.isApprox(Vec3d::Zero()))
|
2019-10-08 11:38:08 +00:00
|
|
|
{
|
|
|
|
m_state.append_translation(translation);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Mouse3DController::handle_packet_rotation(const DataPacket& packet, unsigned int first_byte)
|
|
|
|
{
|
2019-10-09 12:39:28 +00:00
|
|
|
double deadzone = (double)m_state.get_rotation_deadzone();
|
|
|
|
Vec3f rotation(-(float)convert_input(packet[first_byte + 0], packet[first_byte + 1], deadzone),
|
|
|
|
(float)convert_input(packet[first_byte + 2], packet[first_byte + 3], deadzone),
|
|
|
|
-(float)convert_input(packet[first_byte + 4], packet[first_byte + 5], deadzone));
|
2019-10-08 12:32:05 +00:00
|
|
|
|
2019-10-09 12:39:28 +00:00
|
|
|
if (!rotation.isApprox(Vec3f::Zero()))
|
2019-10-08 11:38:08 +00:00
|
|
|
{
|
|
|
|
m_state.append_rotation(rotation);
|
|
|
|
return true;
|
|
|
|
}
|
2019-10-07 07:31:23 +00:00
|
|
|
|
2019-10-08 11:38:08 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Mouse3DController::handle_packet_button(const DataPacket& packet, unsigned int packet_size)
|
|
|
|
{
|
2019-10-09 12:01:13 +00:00
|
|
|
unsigned int data = packet[1] | packet[2] << 8 | packet[3] << 16 | packet[4] << 24;
|
|
|
|
const std::bitset<32> data_bits{ data };
|
|
|
|
for (size_t i = 0; i < data_bits.size(); ++i)
|
2019-10-08 11:38:08 +00:00
|
|
|
{
|
2019-10-09 12:01:13 +00:00
|
|
|
if (data_bits.test(i))
|
2019-10-08 11:38:08 +00:00
|
|
|
{
|
2019-10-09 12:01:13 +00:00
|
|
|
m_state.append_button((unsigned int)i);
|
|
|
|
return true;
|
2019-10-08 11:38:08 +00:00
|
|
|
}
|
2019-09-27 12:52:19 +00:00
|
|
|
}
|
2019-10-08 11:38:08 +00:00
|
|
|
|
|
|
|
return false;
|
2019-09-27 12:52:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|
|
|
|
|
|
|
|
#endif // ENABLE_3DCONNEXION_DEVICES
|