169 lines
9.1 KiB
Markdown
169 lines
9.1 KiB
Markdown
# Teclas
|
|
|
|
> NOTE: Isto não é uma tabela de busca de objetos fornecidos pelo KMK. Esta
|
|
> listagem pode ser encontrada em `keycodes.md`. Possivelmente vale a pena
|
|
> observar o código-fonte bruto se você estiver travado: `kmk/keys.py`.
|
|
|
|
Este é um montante de documentação sobre como teclas físicas são mapeada para
|
|
eventos (e o ciclo-de-vida destes eventos) no KMK. É um tanto técnico, mas se
|
|
você cogita estender a funcionalidade so seu teclado com código extra, você
|
|
precisará de pelo menos uma fração deste conhecimento técnico.
|
|
|
|
Os primeiros passos neste processo não são tão interessantes para a maior parte
|
|
dos fluxos de trabalho, razão pela qual eles estão tão soterrados no KMK:
|
|
varremos um monte de faixas de GPIO (o quão rápido o CircuitPython nos permitir)
|
|
para ver onde, numa matriz de teclas, uma delas foi pressionada. Os detalhes
|
|
técnicos deste processo estão melhor descritos na
|
|
[Wikipedia](https://en.wikipedia.org/wiki/Keyboard_matrix_circuit). Então,
|
|
varremos o mapa de teclas definido, encontrando a primeira tecla válida neste
|
|
índice baseado na pilha de camadas ativas (esta lógica, se você quer ler o
|
|
código, está em `kmk/internal_state.py`, método `_find_key_in_map`).
|
|
|
|
Os próximos passos são a parte interessante, mas para compreendê-los precisamos
|
|
entender um pouco do objeto `Key` (encontrado em `kmk/keys.py`). Objetos `Key`
|
|
têm algumas peças fundamentais de informação:
|
|
|
|
* O `code`, que pode ser qualquer inteiro. Inteiros menores que
|
|
`FIRST_KMK_INTERNAL_KEY` são enviados pela pilha de HID (e portanto para o
|
|
computador, que traduzirá aquele inteiro para algo útil - por exemplo,
|
|
`code=4` torna-se `a` em um teclado US QWERTY/Dvorak)
|
|
|
|
* Os modificadores anexados (para implementar coisas como Shift ou `KC.HYPR` que
|
|
são pressionamentos de teclas singulares enviadas juntamente a mais de uma
|
|
tecla em um único reporte HID. Este é um conceito diferente de Sequências, que
|
|
são uma característica do KMK documentada em `sequences.md`). Para todos os
|
|
propósitos fora do núcleo do KMK, este campo deve ser ignorado - ele pode ser
|
|
seguramente populado mediante meios bem mais sãos que perder tempo fazendo
|
|
isso na mão.
|
|
|
|
* Alguns dados sobre se a tecla deveria ter sido pressionada ou liberada - isto
|
|
é majoritariamente um detalhe de implementação sobre como Sequências
|
|
funcionam, onde, por exemplo, `KC.RALT` pode precisar ser segurada por toda a
|
|
duração da sequência, em vez de ser liberada imediatamente antes de mover para
|
|
o próximo catactere. Geralmente usuários finais não precisam se preocupar com
|
|
isso, mas os campos são denominados `no_press` e `no_release` e são
|
|
referenciados em alguns lugares da base de código se você precisar de
|
|
exemplos.
|
|
|
|
* Manipuladores (*handler*) para o pressionamento (algumas vezes chamado de
|
|
"keydown" ou "press") e liberação (algumas vezes chamado de "keyup" ou
|
|
"release"). KMK fornece manipuladores para funções padrão do yteclado e
|
|
algumas teclas especiais de sobrescrita (como `KC.GESC`, que é uma forma
|
|
aprimorada de teclas ANSI já existentes) em `kmk/handlers/stock.py`, para
|
|
troca de camadas em `kmk/handlers.layers.py`, e para tudo relacionado a
|
|
sequências (veja de novo `sequences.md`) em
|
|
`kmk/handlers/sequences.py`. Discutiremos mais estes em breve.
|
|
|
|
* Chamadas de retorno (*callback*) opcionais a serem executadas antes e/ou
|
|
depois dos handlers acima. Mais sobre isso em breve.
|
|
|
|
* Um campo `meta` genérico, que é mais comumente utilizado para teclas "com
|
|
argumentos" - objetos no objeto `KC` que na realidade são funções que retornam
|
|
instâncias de `Key`, que geralmente precisam acessar os argumentos passados
|
|
para a "função mais externa". Muitos destes exemplos são relacionados com
|
|
trocas de camadas - por exemplo, `KC.MO` é implementada como uma tecla com
|
|
argumentos - quando o usuário acrescenta `KC.MO(1)` ao teclado, a chamada de
|
|
função retorna um objeto `Key` com `meta` contendo um objeto contendo as
|
|
propriedades `layer` e `kc`. Existem outros usos para o campo `meta`, e
|
|
exemplos podem ser encontrados em `kmk/types.py`.
|
|
|
|
Objetos `Key` também podem ser encadeados chamando eles! Para criar uma tecla
|
|
que segura Ctrl e Shift simultaneamente, simplesmente fazemos:
|
|
|
|
```python
|
|
CTRLSHFT = KC.LCTL(KC.LSFT)
|
|
|
|
keyboard.keymap = [ ... CTRLSHFT ... ]
|
|
```
|
|
|
|
Quando uma tecla é pressionada e tiramos um objeto `Key` do keymap, acontecerá o
|
|
seguinte:
|
|
|
|
- Callbacks pré-pressionamento serão executados na ordem em que foram
|
|
atribuídos, com seus valores de retorno descartados (a não ser que o usuário
|
|
os anexe, eles quase nunca existem).
|
|
- O handler de pressionamento correspondente será executado (mais comumente este
|
|
é fornecido pelo KMK).
|
|
- Calllbacks pós-pressionamento serão executados na ordem em que foram
|
|
atribuídos, com seus valores de retorno descartados (a não ser que o usuário
|
|
os anexe, eles quase nunca existem).
|
|
|
|
Os mesmos passos são executados quando uma tecla é liberada.
|
|
|
|
_OK, então... Mas o que é um handler, e o que são esses callbacks?!_
|
|
|
|
Grosso modo, todos eles servem a um mesmo propósiti: fazer algo com os dados da
|
|
tecla, ou executar efeitos colaterais. A maioria dos handlers são fornecidos
|
|
pelo KMK internamente e modificam o `InternalState` de alguma forma -
|
|
adicionando a tecla à fila HID, modificando camadas, etc. Os handlers
|
|
pré/pós-pressionamento são projetados para permitir que a funcionalidade seja
|
|
embutida nestes pontos no fluxo de eventos sem ter que reimplementar (ou
|
|
importar e chamar internamente) os handlers internos.
|
|
|
|
Todos esses métodos recebem os mesmos argumentos, e por isso eu vou copiar a
|
|
docstring direto do código-fonte:
|
|
|
|
> Recebe o seguinte:
|
|
>
|
|
> - self (Esta instância Key)
|
|
> - state (InternalState corrente)
|
|
> - KC (A tabela de busca KC, para conveniência)
|
|
> - `coord_int` (Um inteiro, representação interna da coordenada da matriz para
|
|
> a tecla pressionada - costumeiramente isto não é útil para usuários finais,
|
|
> mas é fornecida por consistência com os manipuladores internos)
|
|
> - `coord_raw` (Uma tupla X,Y de coordenadas da matrix - costumeiramente não é
|
|
> útil, também)
|
|
>
|
|
> O valor de retorno do callback fornecido é descartado. _Exceções não são
|
|
> capturadas_, e provavelmente quebrarão o KMK se não forem tratadas dentro da
|
|
> tua função.
|
|
>
|
|
> Estes handlers são executados na ordem de anexação: handlers fornecidos por
|
|
> chamadas anteriores deste método são executados antes daqueles fornecidos por
|
|
> chamadas posteriores.
|
|
|
|
Isto significa que se você quer adicionar coisas como suporte a LED/brilho, ou
|
|
um botão que aciona seu modem GSM para falar com alguém, ou seja lá o que você
|
|
puder fazer em CircuitPython, que também retenha as capacidades de troca de
|
|
camadas ou seja lá qual for o handler de suporte, você está coberto. Isto também
|
|
significa que você pode adicionar funcionalidades completamente novas ao KMK
|
|
escrevendo seu próprio handler.
|
|
|
|
Eis um exemplo de gancho de ciclo de vida (*lifecycle hook*) que imprime um
|
|
Shrek gigante em Arte ASCII. Ele não utiliza os argumentos que recebe, porque
|
|
não tem intenção de modificar o estado interno. Ele é puramente um efeito
|
|
colateral ([side
|
|
effect](https://en.wikipedia.org/wiki/Side_effect_(computer_science))) executado
|
|
sempre que se pressiona o Alt esquerdo:
|
|
|
|
```python
|
|
def shrek(*args, **kwargs):
|
|
print('⢀⡴⠑⡄⠀⠀⠀⠀⠀⠀⠀⣀⣀⣤⣤⣤⣀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠸⡇⠀⠿⡀⠀⠀⠀⣀⡴⢿⣿⣿⣿⣿⣿⣿⣿⣷⣦⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠑⢄⣠⠾⠁⣀⣄⡈⠙⣿⣿⣿⣿⣿⣿⣿⣿⣆⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⢀⡀⠁⠀⠀⠈⠙⠛⠂⠈⣿⣿⣿⣿⣿⠿⡿⢿⣆⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⢀⡾⣁⣀⠀⠴⠂⠙⣗⡀⠀⢻⣿⣿⠭⢤⣴⣦⣤⣹⠀⠀⠀⢀⢴⣶⣆')
|
|
print('⠀⠀⢀⣾⣿⣿⣿⣷⣮⣽⣾⣿⣥⣴⣿⣿⡿⢂⠔⢚⡿⢿⣿⣦⣴⣾⠁⠸⣼⡿')
|
|
print('⠀⢀⡞⠁⠙⠻⠿⠟⠉⠀⠛⢹⣿⣿⣿⣿⣿⣌⢤⣼⣿⣾⣿⡟⠉⠀⠀⠀⠀⠀')
|
|
print('⠀⣾⣷⣶⠇⠀⠀⣤⣄⣀⡀⠈⠻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠉⠈⠉⠀⠀⢦⡈⢻⣿⣿⣿⣶⣶⣶⣶⣤⣽⡹⣿⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠉⠲⣽⡻⢿⣿⣿⣿⣿⣿⣿⣷⣜⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠀⢸⣿⣿⣷⣶⣮⣭⣽⣿⣿⣿⣿⣿⣿⣿⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠀⠀⣀⣀⣈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠇⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠀⠀⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠃⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠛⠻⠿⠿⠿⠿⠛⠉')
|
|
|
|
|
|
KC.LALT.before_press_handler(shrek)
|
|
```
|
|
|
|
Você também pode copiar uma tecla sem quaisquer handlers pré ou pós nela
|
|
mediante `.clone()`, tal que por exemplo, se eu já adicionei Shrek ao meu `LALT`
|
|
mas quero um `LALT` sem Shrek em algum lugar do keymap, eu posso simplesmente
|
|
clonar a tecla, e a nova tecla não terá meus handlers atrelados a ela:
|
|
|
|
```python
|
|
SHREKLESS_ALT = KC.LALT.clone()
|
|
```
|