summaryrefslogtreecommitdiffstats
path: root/src/input_common/keyboard.cpp
blob: 8261e76fd7782c3fcdb2f989c923a44c63fd1b33 (plain) (blame)
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
// Copyright 2017 Citra Emulator Project
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.

#include <atomic>
#include <list>
#include <mutex>
#include <utility>
#include "input_common/keyboard.h"

namespace InputCommon {

class KeyButton final : public Input::ButtonDevice {
public:
    explicit KeyButton(std::shared_ptr<KeyButtonList> key_button_list_, bool toggle_)
        : key_button_list(std::move(key_button_list_)), toggle(toggle_) {}

    ~KeyButton() override;

    bool GetStatus() const override {
        if (toggle) {
            return toggled_status.load(std::memory_order_relaxed);
        }
        return status.load();
    }

    void ToggleButton() {
        if (lock) {
            return;
        }
        lock = true;
        const bool old_toggle_status = toggled_status.load();
        toggled_status.store(!old_toggle_status);
    }

    void UnlockButton() {
        lock = false;
    }

    friend class KeyButtonList;

private:
    std::shared_ptr<KeyButtonList> key_button_list;
    std::atomic<bool> status{false};
    std::atomic<bool> toggled_status{false};
    bool lock{false};
    const bool toggle;
};

struct KeyButtonPair {
    int key_code;
    KeyButton* key_button;
};

class KeyButtonList {
public:
    void AddKeyButton(int key_code, KeyButton* key_button) {
        std::lock_guard guard{mutex};
        list.push_back(KeyButtonPair{key_code, key_button});
    }

    void RemoveKeyButton(const KeyButton* key_button) {
        std::lock_guard guard{mutex};
        list.remove_if(
            [key_button](const KeyButtonPair& pair) { return pair.key_button == key_button; });
    }

    void ChangeKeyStatus(int key_code, bool pressed) {
        std::lock_guard guard{mutex};
        for (const KeyButtonPair& pair : list) {
            if (pair.key_code == key_code) {
                pair.key_button->status.store(pressed);
                if (pressed) {
                    pair.key_button->ToggleButton();
                } else {
                    pair.key_button->UnlockButton();
                }
                pair.key_button->TriggerOnChange();
            }
        }
    }

    void ChangeAllKeyStatus(bool pressed) {
        std::lock_guard guard{mutex};
        for (const KeyButtonPair& pair : list) {
            pair.key_button->status.store(pressed);
        }
    }

private:
    std::mutex mutex;
    std::list<KeyButtonPair> list;
};

Keyboard::Keyboard() : key_button_list{std::make_shared<KeyButtonList>()} {}

KeyButton::~KeyButton() {
    key_button_list->RemoveKeyButton(this);
}

std::unique_ptr<Input::ButtonDevice> Keyboard::Create(const Common::ParamPackage& params) {
    const int key_code = params.Get("code", 0);
    const bool toggle = params.Get("toggle", false);
    std::unique_ptr<KeyButton> button = std::make_unique<KeyButton>(key_button_list, toggle);
    key_button_list->AddKeyButton(key_code, button.get());
    return button;
}

void Keyboard::PressKey(int key_code) {
    key_button_list->ChangeKeyStatus(key_code, true);
}

void Keyboard::ReleaseKey(int key_code) {
    key_button_list->ChangeKeyStatus(key_code, false);
}

void Keyboard::ReleaseAllKeys() {
    key_button_list->ChangeAllKeyStatus(false);
}

} // namespace InputCommon