mouse_m908/include/mouse_m908.h
2020-05-09 22:33:14 +02:00

319 lines
9.4 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 MOUSE_M908
#define MOUSE_M908
#include <libusb.h>
#include <map>
#include <array>
#include <string>
#include <algorithm>
#include <exception>
#include <regex>
#include <fstream>
#include <sstream>
#include <utility>
#include <iostream>
/**
* The main class representing the M908 mouse.
* This class has member functions to open, close and apply settings to the mouse.
*
* There are 3 main types of functions:
* - set_*: setters
* - get_*: getters
* - write_*: write the settings to the mouse
*
* Therefore the general procedure when changing settings on the mouse is as follows:
* 1. open_mouse() or open_mouse_bus_device()
* 2. set_* (this step only changes the internal state of the class)
* 3. write_* (this step sends the internal state of the class to the mouse)
* 4. close_mouse()
*/
class mouse_m908{
public:
// constructor
mouse_m908();
// enums
/// The available profiles
enum m908_profile{
profile_1 = 0,
profile_2 = 1,
profile_3 = 2,
profile_4 = 3,
profile_5 = 4,
};
/// The available led modes
enum m908_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 m908_report_rate{
r_125Hz,
r_250Hz,
r_500Hz,
r_1000Hz
};
//setter functions
/** Set the currently active profile
*/
int set_profile( m908_profile profile );
/** Set the scrollspeed for the specified profile
* \see _scrollspeed_min
* \see _scrollspeed_max
* \return 0 if successful, 1 if out of bounds
*/
int set_scrollspeed( m908_profile profile, uint8_t speed );
/** Set the led mode for the specified profile
* \see m908_lightmode
* \return 0 if successful
*/
int set_lightmode( m908_profile profile, m908_lightmode lightmode );
/** Set the led color for the specified profile
* \param color color as {r, g, b}
* \return 0 if successful
*/
int set_color( m908_profile profile, std::array<uint8_t, 3> color );
/** Set the led brightness for the specified profile
* \see _brightness_min
* \see _brightness_max
* \return 0 if successful, 1 if out of bounds
*/
int set_brightness( m908_profile profile, uint8_t brightness );
/** Set the led animation speed for the specified profile
* \see _speed_min
* \see _speed_max
* \return 0 if successful, 1 if out of bounds
*/
int set_speed( m908_profile profile, uint8_t speed );
/** Enables/Disables a dpi level for the specified profile
* \see _level_min
* \see _level_max
* \return 0 if successful, 1 if out of bounds
*/
int set_dpi_enable( m908_profile profile, int level, bool enabled );
/** Set the value of a dpi level for the specified profile
* \see _dpi_min
* \see _dpi_max
* \see _level_min
* \see _level_max
* \return 0 if successful, 1 if out of bounds
*/
int set_dpi( m908_profile profile, int level, uint8_t dpi );
/** Set a mapping for a button for the specified profile
* \param mapping 4 bytes for the usb data packets
* \return 0 if successful
*/
int set_key_mapping( m908_profile profile, int key, std::array<uint8_t, 4> mapping );
/** Set a mapping for a button for the specified profile
* \param mapping button name (see keymap.md)
* \return 0 if successful, 1 if mapping is invalid
*/
int set_key_mapping( m908_profile profile, int key, std::string mapping );
/** Set the USB poll rate for the specified profile
* \see m908_report_rate
* \return 0 if successful
*/
int set_report_rate( m908_profile profile, m908_report_rate report_rate );
/** Load the macro from the specified file into the specified slot
* \param macro_number macro slot (1-15)
* \return 0 if successful
*/
int set_macro( int macro_number, std::string file );
/** Set how many times the specified macro should be repeated
* \param macro_number macro slot (1-15)
* \return 0 if successful
*/
int set_macro_repeat( int macro_number, uint8_t repeat );
/** Set whether to try to detach the kernel driver when opening the mouse
*/
int set_detach_kernel_driver( bool detach_kernel_driver );
//getter functions
m908_profile get_profile();
uint8_t get_scrollspeed( m908_profile profile );
m908_lightmode get_lightmode( m908_profile profile );
void get_color( m908_profile profile, std::array<uint8_t, 3> &color );
uint8_t get_brightness( m908_profile profile );
uint8_t get_speed( m908_profile profile );
bool get_dpi_enable( m908_profile profile, int level );
uint8_t get_dpi( m908_profile profile, int level );
m908_report_rate get_report_rate( m908_profile profile );
uint8_t get_macro_repeat( int macro_number );
//writer functions (apply settings to mouse)
/** Write the currently active profile to the mouse
* \return 0 if successful
*/
int write_profile();
/** Write the settings (leds, button mapping, dpi, etc.) to the mouse
* \return 0 if successful
*/
int write_settings();
/** Write a macro to the mouse
* \return 0 if successful
*/
int write_macro( int macro_number );
/** Write the number of repeats for a macro to the mouse
* \return 0 if successful
*/
int write_macro_repeat( int macro_number );
//helper functions
/** Init libusb and open the mouse by its USB VID and PID
* \return 0 if successful
*/
int open_mouse();
/** Init libusb and open the mouse by the USB bus and device adress
* \return 0 if successful
*/
int open_mouse_bus_device( uint8_t bus, uint8_t device );
/** Close the mouse and libusb
* \return 0 if successful (always at the moment)
*/
int close_mouse();
//reader functions (get settings from the mouse)
/**
* Read the settings and print the raw data
*/
int dump_settings();
private:
// whether to detach kernel driver
bool _detach_kernel_driver = true;
//usb device vars
static const uint16_t _mouse_vid;
static const uint16_t _mouse_pid;
libusb_device_handle* _handle;
bool _detached_driver_0 = false;
bool _detached_driver_1 = false;
bool _detached_driver_2 = false;
//setting min and max values
static const uint8_t _scrollspeed_min, _scrollspeed_max;
static const uint8_t _brightness_min, _brightness_max;
static const uint8_t _speed_min, _speed_max;
static const uint8_t _level_min, _level_max;
static const uint8_t _dpi_min, _dpi_max;
//setting vars
m908_profile _profile;
std::array<uint8_t, 5> _scrollspeeds;
std::array<m908_lightmode, 5> _lightmodes;
std::array<std::array<uint8_t, 3>, 5> _colors;
std::array<uint8_t, 5> _brightness_levels;
std::array<uint8_t, 5> _speed_levels;
std::array<std::array<bool, 5>, 5> _dpi_enabled;
std::array<std::array<uint8_t, 5>, 5> _dpi_levels;
std::array<std::array<std::array<uint8_t, 4>, 20>, 5> _keymap_data;
std::array<m908_report_rate, 5> _report_rates;
std::array<std::array<uint8_t, 256>, 15> _macro_data;
std::array<uint8_t, 15> _macro_repeat;
//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> > _keycodes;
/// Values of keyboard modifiers
static const std::map< std::string, uint8_t > _keyboard_modifier_values;
/// Values/keycodes of keyboard keys
static std::map< std::string, uint8_t > _keyboard_key_values;
//usb data packets
/// Used for changing the active profile
static uint8_t _data_profile[6][16];
/// Used for sending the settings, part 1/3
static uint8_t _data_settings_1[15][16];
/// Used for sending the settings, part 2/3
static uint8_t _data_settings_2[64];
/// Used for sending the settings, part 3/3
static uint8_t _data_settings_3[140][16];
/// Used for sending a macro, part 1/3
static uint8_t _data_macros_1[16];
/// Used for sending a macro, part 2/3
static uint8_t _data_macros_2[256];
/// Used for sending a macro, part 3/3
static uint8_t _data_macros_3[16];
/// Lookup table for used when specifying which slot to send a macro to
static uint8_t _data_macros_codes[15][2];
/// Used to send the number repeats for a macro
static uint8_t _data_macros_repeat[16];
/// Used to read the settings, part 1/3
static uint8_t _data_read_1[9][16];
/// Used to read the settings, part 2/3
static uint8_t _data_read_2[85][64];
/// Used to read the settings, part 3/3
static uint8_t _data_read_3[101][16];
};
/* the files are now compiled individually
#include "data.cpp"
#include "constructor.cpp"
#include "helpers.cpp"
#include "getters.cpp"
#include "setters.cpp"
#include "writers.cpp"
*/
#endif