/* * Copyright (C) 2014 Jared Boone, ShareBrained Technology, Inc. * * This file is part of PortaPack. * * 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, 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; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, * Boston, MA 02110-1301, USA. */ #include "debounce.hpp" #include "utility.hpp" uint8_t Debounce::state() { bool v = !pulse_upon_release_ && (state_ || simulated_pulse_); if (simulated_pulse_) simulated_pulse_ = false; return v; } void Debounce::enable_repeat() { repeat_enabled_ = true; } bool Debounce::get_long_press_enabled() const { return long_press_enabled_; } void Debounce::set_long_press_enabled(bool v) { long_press_enabled_ = v; } bool Debounce::long_press_occurred() { bool v = long_press_occurred_; long_press_occurred_ = false; return v; } // Returns TRUE if button state changed (after debouncing) bool Debounce::feed(const uint8_t bit) { history_ = (history_ << 1) | (bit & 1); // "Repeat" handling - simulated button release if (repeat_ctr_) { // Make sure the button is still being held continuously if ((history_ == 0xFF) && !long_press_enabled_) { // Simulate button press every REPEAT_SUBSEQUENT_DELAY ticks if (--repeat_ctr_ == 0) { state_ = !state_; repeat_ctr_ = REPEAT_SUBSEQUENT_DELAY / 2; return true; } } else { // It's a real button release; stop simulating repeat_ctr_ = 0; } } if (state_ == 0) { // Previous button state was 0 (released); // Has button been held for DEBOUNCE_COUNT ticks? if ((history_ & DEBOUNCE_MASK) == DEBOUNCE_MASK) { state_ = 1; held_time_ = 0; // If long_press_enabled_, state() function masks the button press until it's released // or until LONG_PRESS_DELAY is reached if (long_press_enabled_) { pulse_upon_release_ = true; return false; } return true; } } else { // Previous button state was 1 (pressed); // Has button been released for DEBOUNCE_COUNT ticks? if ((history_ & DEBOUNCE_MASK) == 0) { // Button has been released when long_press_enabled_ and before LONG_PRESS_DELAY was reached; // allow state() function to finally return a single press indication (simulated pulse). // Note: In long press mode, apps won't see button press until the button is released. if (pulse_upon_release_) { // force state() function (called by EventDispatcher) to return simulated press for one cycle simulated_pulse_ = true; pulse_upon_release_ = false; } else { state_ = 0; } // Reset long_press_occurred_ flag after button is released long_press_occurred_ = false; return true; } // Has button been held continuously? if (history_ == 0xFF) { held_time_++; if (pulse_upon_release_) { // Button is being held down and long_press support is enabled for this key: // if LONG_PRESS_DELAY is reached then finally report that switch is pressed and set flag // indicating it was a LONG press // (note that repeat_support and long_press support are mutually exclusive) if (held_time_ >= LONG_PRESS_DELAY) { long_press_occurred_ = true; simulated_pulse_ = true; pulse_upon_release_ = false; held_time_ = 0; return true; } } else if (repeat_enabled_ && !long_press_enabled_) { // Repeat support -- 4 directional buttons only (unless long_press is enabled) if (held_time_ == REPEAT_INITIAL_DELAY) { // Delay reached; trigger repeat code on NEXT tick repeat_ctr_ = 1; held_time_ = 0; } } } else { // Button not continuously pressed; reset counter held_time_ = 0; } } return false; }