summaryrefslogtreecommitdiffstats
path: root/keyboards/cipulot/ec_23u/keymaps/via/via_apc.c
blob: 5ea77af44c8bd2ecf29c4368e10d283f5b83fa68 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
/* Copyright 2023 Cipulot
 *
 * 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 2 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, see <http://www.gnu.org/licenses/>.
 */

#include "ec_switch_matrix.h"
#include "action.h"
#include "via.h"

void apc_init_thresholds(void);
void apc_set_threshold(bool is_for_actuation);

// Declaring an _apc_config_t struct that will store our data
typedef struct _apc_config_t {
    uint16_t actuation_threshold;
    uint16_t release_threshold;
} apc_config;

// Check if the size of the reserved persistent memory is the same as the size of struct apc_config
_Static_assert(sizeof(apc_config) == EECONFIG_USER_DATA_SIZE, "Mismatch in keyboard EECONFIG stored data");

// Declaring a new variable apc of type apc_config
apc_config apc;

// Declaring enums for VIA config menu
enum via_apc_enums {
    // clang-format off
    id_apc_actuation_threshold = 1,
    id_apc_release_threshold = 2
    // clang-format on
};

// Initializing persistent memory configuration: default values are declared and stored in PMEM
void eeconfig_init_user(void) {
    // Default values
    apc.actuation_threshold = DEFAULT_ACTUATION_LEVEL;
    apc.release_threshold   = DEFAULT_RELEASE_LEVEL;
    // Write default value to EEPROM now
    eeconfig_update_user_datablock(&apc);
}

// On Keyboard startup
void keyboard_post_init_user(void) {
    // Read custom menu variables from memory
    eeconfig_read_user_datablock(&apc);
    apc_init_thresholds();
}

// Handle the data received by the keyboard from the VIA menus
void apc_config_set_value(uint8_t *data) {
    // data = [ value_id, value_data ]
    uint8_t *value_id   = &(data[0]);
    uint8_t *value_data = &(data[1]);

    switch (*value_id) {
        case id_apc_actuation_threshold: {
            apc.actuation_threshold = value_data[1] | (value_data[0] << 8);
            apc_set_threshold(true);
            break;
        }
        case id_apc_release_threshold: {
            apc.release_threshold = value_data[1] | (value_data[0] << 8);
            apc_set_threshold(false);
            break;
        }
    }
}

// Handle the data sent by the keyboard to the VIA menus
void apc_config_get_value(uint8_t *data) {
    // data = [ value_id, value_data ]
    uint8_t *value_id   = &(data[0]);
    uint8_t *value_data = &(data[1]);

    switch (*value_id) {
        case id_apc_actuation_threshold: {
            value_data[0] = apc.actuation_threshold >> 8;
            value_data[1] = apc.actuation_threshold & 0xFF;
            break;
        }
        case id_apc_release_threshold: {
            value_data[0] = apc.release_threshold >> 8;
            value_data[1] = apc.release_threshold & 0xFF;
            break;
        }
    }
}

// Save the data to persistent memory after changes are made
void apc_config_save(void) {
    eeconfig_update_user_datablock(&apc);
}

void via_custom_value_command_kb(uint8_t *data, uint8_t length) {
    // data = [ command_id, channel_id, value_id, value_data ]
    uint8_t *command_id        = &(data[0]);
    uint8_t *channel_id        = &(data[1]);
    uint8_t *value_id_and_data = &(data[2]);

    if (*channel_id == id_custom_channel) {
        switch (*command_id) {
            case id_custom_set_value: {
                apc_config_set_value(value_id_and_data);
                break;
            }
            case id_custom_get_value: {
                apc_config_get_value(value_id_and_data);
                break;
            }
            case id_custom_save: {
                apc_config_save();
                break;
            }
            default: {
                // Unhandled message.
                *command_id = id_unhandled;
                break;
            }
        }
        return;
    }

    *command_id = id_unhandled;
}

// Initialize the thresholds
void apc_init_thresholds(void) {
    ecsm_config.ecsm_actuation_threshold = apc.actuation_threshold;
    ecsm_config.ecsm_release_threshold   = apc.release_threshold;

    // Update the ecsm_config
    ecsm_update(&ecsm_config);
}

// Set the thresholds
void apc_set_threshold(bool is_for_actuation) {
    if (is_for_actuation) {
        ecsm_config.ecsm_actuation_threshold = apc.actuation_threshold;

    } else {
        ecsm_config.ecsm_release_threshold = apc.release_threshold;
    }
    // Update the ecsm_config
    ecsm_update(&ecsm_config);
}