mouse_m908/include/m990/setters.cpp
2020-11-12 15:10:59 +01:00

251 lines
6.1 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.
*
*/
#include "mouse_m990.h"
//setter functions
int mouse_m990::set_profile( rd_profile profile ){
_s_profile = profile;
return 0;
}
int mouse_m990::set_scrollspeed( rd_profile profile, uint8_t speed ){
//check if bounds exceeded
if( speed < _c_scrollspeed_min || speed > _c_scrollspeed_max ){
return 1;
}
_s_scrollspeeds[profile] = speed;
return 0;
}
int mouse_m990::set_lightmode( rd_profile profile, rd_lightmode lightmode ){
_s_lightmodes[profile] = lightmode;
return 0;
}
int mouse_m990::set_color( rd_profile profile, std::array<uint8_t, 3> color ){
_s_colors[profile] = color;
return 0;
}
int mouse_m990::set_brightness( rd_profile profile, uint8_t brightness ){
//check bounds
if( brightness < _c_brightness_min || brightness > _c_brightness_max ){
return 1;
}
_s_brightness_levels[profile] = brightness;
return 0;
}
int mouse_m990::set_speed( rd_profile profile, uint8_t speed ){
//check bounds
if( speed < _c_speed_min || speed > _c_speed_max ){
return 1;
}
_s_speed_levels[profile] = speed;
return 0;
}
int mouse_m990::set_dpi_enable( rd_profile profile, int level, bool enabled ){
//check bounds
if( level < _c_level_min || level > _c_level_max ){
return 1;
}
_s_dpi_enabled[profile][level] = enabled;
// check if at least one level enabled
int sum = 0;
for( int i = _c_level_min; i <= _c_level_max; i++ ){
if( _s_dpi_enabled[profile][i] ){
sum++;
}
}
// if no level enabled: reenable specified level
if( sum == 0 ){
_s_dpi_enabled[profile][level] = true;
return 1;
}
return 0;
}
int mouse_m990::set_dpi( rd_profile profile, int level, std::string dpi ){
// check DPI level bounds
if( level < _c_level_min || level > _c_level_max )
return 1;
// check format: 0xABCD (raw bytes)
if( std::regex_match( dpi, std::regex("0x[[:xdigit:]]{4}") ) ){
uint8_t b0 = (uint8_t)stoi( dpi.substr(2,2), 0, 16 );
uint8_t b1 = (uint8_t)stoi( dpi.substr(4,2), 0, 16 );
//check bounds
if( b0 < _c_dpi_min || b0 > _c_dpi_max || b1 < _c_dpi_2_min || b1 > _c_dpi_2_max )
return 1;
_s_dpi_levels[profile][level][0] = b0;
_s_dpi_levels[profile][level][1] = b1;
return 0;
}
// check format: 1234 (real DPI)
else if( std::regex_match( dpi, std::regex("[[:digit:]]+") ) ){
if( _c_dpi_codes.find( std::stoi(dpi) ) != _c_dpi_codes.end() ){
_s_dpi_levels[profile][level][0] = _c_dpi_codes.at( std::stoi(dpi) )[0];
_s_dpi_levels[profile][level][1] = _c_dpi_codes.at( std::stoi(dpi) )[1];
return 0;
}
}
return 1;
}
int mouse_m990::set_dpi( rd_profile profile, int level, std::array<uint8_t, 2> dpi ){
// check DPI level bounds
if( level < _c_level_min || level > _c_level_max )
return 1;
//check bounds
if( dpi[0] < _c_dpi_min || dpi[0] > _c_dpi_max || dpi[1] < _c_dpi_2_min || dpi[1] > _c_dpi_2_max )
return 1;
_s_dpi_levels[profile][level][0] = dpi[0];
_s_dpi_levels[profile][level][1] = dpi[1];
return 0;
}
int mouse_m990::set_key_mapping( rd_profile profile, int key, std::array<uint8_t, 4> mapping ){
_s_keymap_data[profile][key][0] = mapping[0];
_s_keymap_data[profile][key][1] = mapping[1];
_s_keymap_data[profile][key][2] = mapping[2];
_s_keymap_data[profile][key][3] = mapping[3];
return 0;
}
int mouse_m990::set_key_mapping( rd_profile profile, int key, std::string mapping ){
// valid key ?
if( _c_button_names[key] == "" )
return 1;
std::array< uint8_t, 4 > bytes;
int return_value = _i_encode_button_mapping( mapping, bytes );
if( return_value == 0 ){
_s_keymap_data[profile][key][0] = bytes[0];
_s_keymap_data[profile][key][1] = bytes[1];
_s_keymap_data[profile][key][2] = bytes[2];
_s_keymap_data[profile][key][3] = bytes[3];
}
return return_value;
}
int mouse_m990::set_report_rate( rd_profile profile, rd_report_rate report_rate ){
_s_report_rates[profile] = report_rate;
return 0;
}
int mouse_m990::set_macro( int macro_number, std::string file ){
//check if macro_number is valid
if( macro_number < 1 || macro_number > 15 ){
return 1;
}
//open file
std::ifstream config_in( file );
if( !config_in.is_open() ){
return 1;
}
std::array< uint8_t, 256 > macro_bytes;
_i_encode_macro( macro_bytes, config_in, 8 );
std::copy( macro_bytes.begin()+8, macro_bytes.end(), _s_macro_data[macro_number-1].begin()+8 );
config_in.close();
return 0;
}
int mouse_m990::set_all_macros( std::string file ){
//open file
std::ifstream config_in( file );
if( !config_in.is_open() ){
return 1;
}
int macro_number = 0; // initially invalid
std::array< std::stringstream, 15 > macro_streams;
// get all macros from file
for( std::string line; std::getline(config_in, line); ){
// empty line → skip
if( line.length() == 0 )
continue;
// macro header → set macro_number
if( std::regex_match( line, std::regex(";## macro[0-9]*") ) ){
macro_number = stoi( std::regex_replace( line, std::regex(";## macro"), "" ), 0, 10 );
}
// macro action
if( std::regex_match( line, std::regex(";# .*") ) ){
// valid macronumber ?
if( macro_number < 1 || macro_number > 15 )
continue;
macro_streams.at( macro_number-1 ) << std::regex_replace( line, std::regex(";# "), "" ) << "\n";
}
}
// encode and store all macros
for( int i = 0; i < 15; i++ ){
std::array< uint8_t, 256 > macro_bytes;
_i_encode_macro( macro_bytes, macro_streams.at(i), 8 );
std::copy( macro_bytes.begin()+8, macro_bytes.end(), _s_macro_data.at(i).begin()+8 );
}
return 0;
}