171 lines
5.2 KiB
C++
171 lines
5.2 KiB
C++
/*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
* MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
//main class
|
|
#ifndef RD_MOUSE
|
|
#define RD_MOUSE
|
|
|
|
#include <libusb.h>
|
|
#include <map>
|
|
#include <array>
|
|
#include <string>
|
|
#include <algorithm>
|
|
#include <exception>
|
|
#include <regex>
|
|
#include <fstream>
|
|
#include <sstream>
|
|
#include <utility>
|
|
#include <iostream>
|
|
#include <iomanip>
|
|
#include <string>
|
|
|
|
/**
|
|
* This class is used as a base for the different models
|
|
*
|
|
* Global constants (e.g. keyboard keycodes) and universal functions
|
|
* should be placed in this class.
|
|
*
|
|
*/
|
|
class rd_mouse{
|
|
|
|
public:
|
|
|
|
// enums
|
|
/// The available profiles
|
|
enum rd_profile{
|
|
profile_1 = 0,
|
|
profile_2 = 1,
|
|
profile_3 = 2,
|
|
profile_4 = 3,
|
|
profile_5 = 4,
|
|
};
|
|
|
|
/// The available led modes
|
|
enum rd_lightmode{
|
|
lightmode_breathing,
|
|
lightmode_rainbow,
|
|
lightmode_static,
|
|
lightmode_wave,
|
|
lightmode_alternating,
|
|
lightmode_reactive,
|
|
lightmode_flashing,
|
|
lightmode_off,
|
|
};
|
|
|
|
/// The available USB report rates (polling rates)
|
|
enum rd_report_rate{
|
|
r_125Hz,
|
|
r_250Hz,
|
|
r_500Hz,
|
|
r_1000Hz
|
|
};
|
|
|
|
/** \brief Detects supported mice
|
|
* In the case of multiple connected mice, only the first will be detected
|
|
* \return The model name of the mouse, empty string if no mouse was found
|
|
*/
|
|
static std::string detect();
|
|
|
|
/// Set whether to try to detach the kernel driver when opening the mouse
|
|
void set_detach_kernel_driver( bool detach_kernel_driver ){
|
|
_i_detach_kernel_driver = detach_kernel_driver;
|
|
}
|
|
/// Get _i_detach_kernel_driver
|
|
bool get_detach_kernel_driver(){ return _i_detach_kernel_driver; }
|
|
|
|
protected:
|
|
|
|
//setting min and max values
|
|
static const uint8_t _c_scrollspeed_min, _c_scrollspeed_max;
|
|
static const uint8_t _c_brightness_min, _c_brightness_max;
|
|
static const uint8_t _c_speed_min, _c_speed_max;
|
|
static const uint8_t _c_level_min, _c_level_max;
|
|
static const uint8_t _c_dpi_min, _c_dpi_max;
|
|
static const uint8_t _c_dpi_2_min, _c_dpi_2_max;
|
|
|
|
//mapping of button names to values
|
|
/// Values/keycodes of mouse buttons and special button functions
|
|
static std::map< std::string, std::array<uint8_t, 3> > _c_keycodes;
|
|
/// Values of keyboard modifiers
|
|
static const std::map< std::string, uint8_t > _c_keyboard_modifier_values;
|
|
/// Values/keycodes of keyboard keys
|
|
static std::map< std::string, uint8_t > _c_keyboard_key_values;
|
|
/// DPI values for the snipe button
|
|
static std::map< int, uint8_t > _c_snipe_dpi_values;
|
|
|
|
//usb device handling
|
|
/// libusb device handle
|
|
libusb_device_handle* _i_handle;
|
|
/// whether to detach kernel driver
|
|
bool _i_detach_kernel_driver = true;
|
|
/// set by open_mouse for close_mouse
|
|
bool _i_detached_driver_0 = false;
|
|
/// set by open_mouse for close_mouse
|
|
bool _i_detached_driver_1 = false;
|
|
/// set by open_mouse for close_mouse
|
|
bool _i_detached_driver_2 = false;
|
|
|
|
/** \brief Init libusb and open the mouse by its USB VID and PID
|
|
* \return 0 if successful
|
|
*/
|
|
int _i_open_mouse( const uint16_t vid, const uint16_t pid );
|
|
|
|
/** \brief Init libusb and open the mouse by the USB bus and device adress
|
|
* \return 0 if successful
|
|
*/
|
|
int _i_open_mouse_bus_device( const uint8_t bus, const uint8_t device );
|
|
|
|
/** \brief Close the mouse and libusb
|
|
* \return 0 if successful (always at the moment)
|
|
*/
|
|
int _i_close_mouse();
|
|
|
|
|
|
/** \brief Decode macro byte code (of one macro) and print the commands to output
|
|
* \arg macro_bytes the macro bytecode
|
|
* \arg output where to print to
|
|
* \arg prefix string printed before each action
|
|
* \arg offset start decoding from macro_bytes[offset], set to 0 if offset >= macro_bytes.size()
|
|
* \return 0 if no invalid codes were encountered
|
|
*/
|
|
int _i_decode_macro( std::vector< uint8_t >& macro_bytes, std::ostream& output, std::string prefix, size_t offset );
|
|
|
|
/** \brief Decodes the bytes describing a button mapping
|
|
* \arg bytes the 4 bytes descriping the mapping
|
|
* \arg mapping string to hold the result
|
|
* \return 0 if valid button mapping
|
|
* \see _i_encode_button_mapping
|
|
*/
|
|
int _i_decode_button_mapping( std::array<uint8_t, 4>& bytes, std::string& mapping );
|
|
|
|
/** \brief Turns a string describing a button mapping into bytecode
|
|
* \arg mapping button mapping
|
|
* \arg bytes holds the result
|
|
* \return 0 if valid button mapping
|
|
* \see _i_decode_button_mapping
|
|
*/
|
|
int _i_encode_button_mapping( std::string& mapping, std::array<uint8_t, 4>& bytes );
|
|
};
|
|
|
|
// include header files for the individual models
|
|
#include "m908/mouse_m908.h"
|
|
#include "m709/mouse_m709.h"
|
|
#include "m711/mouse_m711.h"
|
|
#include "m715/mouse_m715.h"
|
|
|
|
#endif
|