kmk_firmware/docs/ptBR/keys.md
AndersonTorres 02a9a088a0 Small fixups
2021-08-30 11:16:55 -07:00

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()
```