Skip to content

Commit f833ebb

Browse files
authored
Merge pull request #1706 from contour-terminal/fix/repeatable_actions
Make some actions repeatable
2 parents bf0a315 + 985b0c6 commit f833ebb

File tree

3 files changed

+31
-4
lines changed

3 files changed

+31
-4
lines changed

src/contour/Actions.h

+4
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
#pragma once
33

44
#include <crispy/assert.h>
5+
#include <crispy/utils.h>
56

67
#include <format>
78
#include <optional>
@@ -151,6 +152,9 @@ using Action = std::variant<CancelSelection,
151152
SwitchToTabLeft,
152153
SwitchToTabRight>;
153154

155+
template <typename T>
156+
concept NonRepeatableActionConcept = crispy::one_of<T, CreateNewTab, CloseTab>;
157+
154158
std::optional<Action> fromString(std::string const& name);
155159

156160
namespace documentation

src/contour/TerminalSession.cpp

+24-4
Original file line numberDiff line numberDiff line change
@@ -814,6 +814,28 @@ void TerminalSession::onScrollOffsetChanged(vtbackend::ScrollOffset value)
814814
}
815815
// }}}
816816
// {{{ Input Events
817+
818+
void handleAction(auto const& actions, auto eventType, auto callback)
819+
{
820+
if (eventType == KeyboardEventType::Press)
821+
callback(*actions);
822+
else if (eventType == KeyboardEventType::Repeat)
823+
{
824+
// filter out actions that are not repeatable
825+
std::vector<actions::Action> tmpActions;
826+
auto set = crispy::overloaded {
827+
[&]([[maybe_unused]] actions::NonRepeatableActionConcept auto const& action) {},
828+
[&](auto const& action) { tmpActions.emplace_back(action); },
829+
};
830+
831+
for (auto const& action: *actions)
832+
{
833+
std::visit(set, action);
834+
}
835+
callback(tmpActions);
836+
}
837+
}
838+
817839
void TerminalSession::sendKeyEvent(Key key, Modifiers modifiers, KeyboardEventType eventType, Timestamp now)
818840
{
819841
inputLog()("Key {} event received: {} {}", eventType, modifiers, key);
@@ -833,8 +855,7 @@ void TerminalSession::sendKeyEvent(Key key, Modifiers modifiers, KeyboardEventTy
833855
if (auto const* actions =
834856
config::apply(_config.inputMappings.value().keyMappings, key, modifiers, matchModeFlags()))
835857
{
836-
if (eventType == KeyboardEventType::Press)
837-
executeAllActions(*actions);
858+
handleAction(actions, eventType, [&](auto const& actions) { executeAllActions(actions); });
838859
return;
839860
}
840861
}
@@ -869,8 +890,7 @@ void TerminalSession::sendCharEvent(
869890
config::apply(_config.inputMappings.value().charMappings, value, modifiers, matchModeFlags());
870891
actions && !_terminal.inputHandler().isEditingSearch())
871892
{
872-
if (eventType == KeyboardEventType::Press)
873-
executeAllActions(*actions);
893+
handleAction(actions, eventType, [&](auto const& actions) { executeAllActions(actions); });
874894
return;
875895
}
876896
}

src/crispy/utils.h

+3
Original file line numberDiff line numberDiff line change
@@ -560,4 +560,7 @@ struct overloaded: Ts...
560560
template <class... Ts>
561561
overloaded(Ts...) -> overloaded<Ts...>;
562562

563+
template <typename T, typename... Ts>
564+
concept one_of = (std::same_as<T, Ts> || ...);
565+
563566
} // namespace crispy

0 commit comments

Comments
 (0)