qmk_firmware/static/Features/Stenography.html

413 lines
24 KiB
HTML
Raw Normal View History

2018-05-06 19:34:47 +00:00
<!DOCTYPE html>
<html class="no-js" lang="en">
<head>
<title>Stenography - QMK Firmware</title>
<meta name="description" content="Keyboard controller firmware for Atmel AVR and ARM USB families">
<meta name="author" content="QMK Community">
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
2018-05-08 04:02:24 +00:00
<link rel="icon" href="../themes/daux/img/favicon-navy.png" type="image/x-icon">
2018-05-06 19:34:47 +00:00
<!-- Mobile -->
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<!-- Font -->
<!-- CSS -->
2018-05-08 04:02:24 +00:00
<link href='../themes/daux/css/theme-navy.min.css' rel='stylesheet' type='text/css'>
2018-05-06 19:34:47 +00:00
<!-- Tipue Search -->
<link href="../tipuesearch/tipuesearch.css" rel="stylesheet">
<!--[if lt IE 9]>
<script src="../themes/daux/js/html5shiv-3.7.3.min.js"></script>
<![endif]-->
</head>
<body class=" ">
<div class="Columns content">
<aside class="Columns__left Collapsible">
<button type="button" class="Button Collapsible__trigger">
<span class="Collapsible__trigger__bar"></span>
<span class="Collapsible__trigger__bar"></span>
<span class="Collapsible__trigger__bar"></span>
</button>
<a class="Brand" href="../index.html">QMK Firmware</a>
<div class="Search">
<svg class="Search__icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 451 451">
<path d="M447.05 428l-109.6-109.6c29.4-33.8 47.2-77.9 47.2-126.1C384.65 86.2 298.35 0 192.35 0 86.25 0 .05 86.3.05 192.3s86.3 192.3 192.3 192.3c48.2 0 92.3-17.8 126.1-47.2L428.05 447c2.6 2.6 6.1 4 9.5 4s6.9-1.3 9.5-4c5.2-5.2 5.2-13.8 0-19zM26.95 192.3c0-91.2 74.2-165.3 165.3-165.3 91.2 0 165.3 74.2 165.3 165.3s-74.1 165.4-165.3 165.4c-91.1 0-165.3-74.2-165.3-165.4z"/>
</svg>
<input type="search" id="tipue_search_input" class="Search__field" placeholder="Search..." autocomplete="on"
results=25 autosave=text_search>
</div>
<div class="Collapsible__content">
<!-- Navigation -->
2018-05-08 04:02:24 +00:00
<ul class='Nav'><li class='Nav__item has-children'><a href="../Getting_Started/index.html" class="folder"><i class="Nav__arrow">&nbsp;</i>Getting Started</a><ul class='Nav'><li class='Nav__item has-children'><a href="../Getting_Started/Install_Build_Tools/index.html" class="folder"><i class="Nav__arrow">&nbsp;</i>Install Build Tools</a><ul class='Nav'><li class='Nav__item '><a href="../Getting_Started/Install_Build_Tools/Vagrant.html">Vagrant</a></li></ul></li><li class='Nav__item '><a href="../Getting_Started/Build_Compile_Instructions.html">Build Compile Instructions</a></li><li class='Nav__item '><a href="../Getting_Started/Flashing_Firmware.html">Flashing Firmware</a></li><li class='Nav__item '><a href="../Getting_Started/Contributing.html">Contributing</a></li><li class='Nav__item '><a href="../Getting_Started/How_to_Use_GitHub.html">How to Use GitHub</a></li><li class='Nav__item '><a href="../Getting_Started/Getting_Help.html">Getting Help</a></li></ul></li><li class='Nav__item has-children'><a href="../Complete_Newbs_Guide/index.html" class="folder"><i class="Nav__arrow">&nbsp;</i>Complete Newbs Guide</a><ul class='Nav'><li class='Nav__item '><a href="../Complete_Newbs_Guide/Complete_Newbie's_Guide.html">Complete Newbie's Guide</a></li><li class='Nav__item '><a href="../Complete_Newbs_Guide/Building_Your_First_Firmware.html">Building Your First Firmware</a></li><li class='Nav__item '><a href="../Complete_Newbs_Guide/Flashing_Firmware.html">Flashing Firmware</a></li><li class='Nav__item '><a href="../Complete_Newbs_Guide/Testing_and_Debugging.html">Testing and Debugging</a></li></ul></li><li class='Nav__item has-children'><a href="../FAQ/index.html" class="folder"><i class="Nav__arrow">&nbsp;</i>FAQ</a><ul class='Nav'><li class='Nav__item '><a href="../FAQ/General_FAQ.html">General FAQ</a></li><li class='Nav__item '><a href="../FAQ/Build_Compile_QMK.html">Build Compile QMK</a></li><li class='Nav__item '><a href="../FAQ/Debugging_and_Troubleshooting.html">Debugging and Troubleshooting</a></li><li class='Nav__item '><a href="../FAQ/Keymaps.html">Keymaps</a></li></ul></li><li class='Nav__item has-children'><a href="../Hardware/index.html" class="folder"><i class="Nav__arrow">&nbsp;</i>Hardware</a><ul class='Nav'><li class='Nav__item '><a href="../Hardware/AVR_Processors.html">AVR Processors</a></li><li class='Nav__item '><a href="../Hardware/Drivers.html">Drivers</a></li></ul></li><li class='Nav__item Nav__item--open has-children'><a href="../Features/index.html" class="folder"><i class="Nav__arrow">&nbsp;</i>Features</a><ul class='Nav'><li class='Nav__item '><a href="../Features/Advanced_Keycodes.html">Advanced Keycodes</a></li><li class='Nav__item '><a href="../Features/Audio.html">Audio</a></li><li class='Nav__item '><a href="../Features/Auto_Shift.html">Auto Shift</a></li><li class='Nav__item '><a href="../Features/Backlight.html">Backlight</a></li><li class='Nav__item '><a href="../Features/Bluetooth.html">Bluetooth</a></li><li class='Nav__item '><a href="../Features/Bootmagic.html">Bootmagic</a></li><li class='Nav__item '><a href="../Features/Command.html">Command</a></li><li class='Nav__item '><a href="../Features/Dynamic_Macros.html">Dynamic Macros</a></li><li class='Nav__item '><a href="../Features/Grave_Escape.html">Grave Escape</a></li><li class='Nav__item '><a href="../Features/Key_Lock.html">Key Lock</a></li><li class='Nav__item '><a href="../Features/Layouts.html">Layouts</a></li><li class='Nav__item '><a href="../Features/Leader_Key.html">Leader Key</a></li><li class='Nav__item '><a href="../Features/Macros.html">Macros</a></li><li class='Nav__item '><a href="../Features/Mouse_Keys.html">Mouse Keys</a></li><li class='Nav__item '><a href="../Features/Pointing_Device.html">Pointing Device</a></li><li class='Nav__item '><a href="../Features/PS_2_Mouse.html">PS 2 Mouse</a></li><li class='Nav__item '><a href="../Features/RGB_Lighting.html">RGB Lighting</a></li><li class='Nav__item '><a href="../Features/Space_Cadet_Shift.html">Space Cadet Shift</a></li><li class='Nav__item '><a href="../Fea
2018-05-06 19:34:47 +00:00
<div class="Links">
2018-05-08 04:02:24 +00:00
</div>
2018-05-06 19:34:47 +00:00
</div>
</aside>
<div class="Columns__right Columns__right--full">
<div class="Columns__right__content">
<div class="doc_content">
<article class="Page">
<div class="Page__header">
<h1><a href="../Features/index.html">Features</a> <svg class="Page__header--separator" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 477.175 477.175"><path d="M360.73 229.075l-225.1-225.1c-5.3-5.3-13.8-5.3-19.1 0s-5.3 13.8 0 19.1l215.5 215.5-215.5 215.5c-5.3 5.3-5.3 13.8 0 19.1 2.6 2.6 6.1 4 9.5 4 3.4 0 6.9-1.3 9.5-4l225.1-225.1c5.3-5.2 5.3-13.8.1-19z"/></svg> <a href="../Features/Stenography.html">Stenography</a></h1>
2018-05-08 04:02:24 +00:00
<span class="EditOn">
<a href="https://github.com/qmk/qmk_firmware/blob/master/docs/05_Features/Stenography.md" target="_blank">
Edit on GitHub </a>
</span>
</div>
2018-05-06 19:34:47 +00:00
<div class="s-content">
<ul class="TableOfContents">
<li>
<p><a href="#page_Stenography-in-QMK">Stenography in QMK</a></p>
<ul class="TableOfContents">
<li>
<p><a href="#page_Plover-with-QWERTY-Keyboard">Plover with QWERTY Keyboard</a></p>
</li>
<li>
<p><a href="#page_Plover-with-Steno-Protocol">Plover with Steno Protocol</a></p>
<ul class="TableOfContents">
<li>
<p><a href="#page_TX-Bolt">TX Bolt</a></p>
</li>
<li>
<p><a href="#page_GeminiPR">GeminiPR</a></p>
</li>
</ul>
</li>
<li>
<p><a href="#page_Configuring-QMK-for-Steno">Configuring QMK for Steno</a></p>
</li>
<li>
<p><a href="#page_Learning-Stenography">Learning Stenography</a></p>
</li>
<li>
<p><a href="#page_Interfacing-with-the-code">Interfacing with the code</a></p>
</li>
<li>
<p><a href="#page_Keycode-Reference">Keycode Reference</a></p>
</li>
</ul>
</li>
</ul>
<h1 id="page_Stenography-in-QMK">Stenography in QMK</h1>
<p><a href="https://en.wikipedia.org/wiki/Stenotype" class="Link--external">Stenography</a> is a method of writing most often used by court reports, closed-captioning, and real-time transcription for the deaf. In stenography words are chorded syllable by syllable with a mixture of spelling, phonetic, and shortcut (briefs) strokes. Professional stenographers can reach 200-300 WPM without any of the strain usually found in standard typing and with far fewer errors (&gt;99.9% accuracy).</p>
<p>The <a href="http://www.openstenoproject.org/" class="Link--external">Open Steno Project</a> has built an open-source program called Plover that provides real-time translation of steno strokes into words and commands. It has an established dictionary and supports</p>
<h2 id="page_Plover-with-QWERTY-Keyboard">Plover with QWERTY Keyboard</h2>
<p>Plover can work with any standard QWERTY keyboard, although it is more efficient if the keyboard supports NKRO (n-key rollover) to allow Plover to see all the pressed keys at once. An example keymap for Plover can be found in <code>planck/keymaps/default</code>. Switching to the <code>PLOVER</code> layer adjusts the position of the keyboard to support the number bar.</p>
<p>To use Plover with QMK just enable NKRO and optionally adjust your layout if you have anything other than a standard layout. You may also want to purchase some steno-friendly keycaps to make it easier to hit multiple keys.</p>
<h2 id="page_Plover-with-Steno-Protocol">Plover with Steno Protocol</h2>
<p>Plover also understands the language of several steno machines. QMK can speak a couple of these languages, TX Bolt and GeminiPR. An example layout can be found in <code>planck/keymaps/steno</code>.</p>
<p>When QMK speaks to Plover over a steno protocol Plover will not use the keyboard as input. This means that you can switch back and forth between a standard keyboard and your steno keyboard, or even switch layers from Plover to standard and back without needing to activate/deactivate Plover.</p>
<p>In this mode Plover expects to speak with a steno machine over a serial port so QMK will present itself to the operating system as a virtual serial port in addition to a keyboard. By default QMK will speak the TX Bolt protocol but can be switched to GeminiPR; the last protocol used is stored in non-volatile memory so QMK will use the same protocol on restart.</p>
<blockquote>
<p>Note: Due to hardware limitations you may not be able to run both a virtual serial port and mouse emulation at the same time.</p>
</blockquote>
<h3 id="page_TX-Bolt">TX Bolt</h3>
<p>TX Bolt communicates the status of 24 keys over a very simple protocol in variable-sized (1-5 byte) packets.</p>
<h3 id="page_GeminiPR">GeminiPR</h3>
<p>GeminiPR encodes 42 keys into a 6-byte packet. While TX Bolt contains everything that is necessary for standard stenography, GeminiPR opens up many more options, including supporting non-English theories.</p>
<h2 id="page_Configuring-QMK-for-Steno">Configuring QMK for Steno</h2>
<p>Firstly, enable steno in your keymap's Makefile. You may also need disable mousekeys, extra keys, or another USB endpoint to prevent conflicts. The builtin USB stack for some processors only supports a certain number of USB endpoints and the virtual serial port needed for steno fills 3 of them.</p>
<pre><code class="language-Makefile">STENO_ENABLE = yes
MOUSEKEY_ENABLE = no
</code></pre>
<p>In your keymap create a new layer for Plover. You will need to include <code>keymap_steno.h</code>. See <code>planck/keymaps/steno/keymap.c</code> for an example. Remember to create a key to switch to the layer as well as a key for exiting the layer. If you would like to switch modes on the fly you can use the keycodes <code>QK_STENO_BOLT</code> and <code>QK_STENO_GEMINI</code>. If you only want to use one of the protocols you may set it up in your initialization function:</p>
<pre><code class="language-C">void matrix_init_user() {
steno_set_mode(STENO_MODE_GEMINI); // or STENO_MODE_BOLT
}
</code></pre>
<p>Once you have your keyboard flashed launch Plover. Click the 'Configure...' button. In the 'Machine' tab select the Stenotype Machine that corresponds to your desired protocol. Click the 'Configure...' button on this tab and enter the serial port or click 'Scan'. Baud rate is fine at 9600 (although you should be able to set as high as 115200 with no issues). Use the default settings for everything else (Data Bits: 8, Stop Bits: 1, Parity: N, no flow control).</p>
<p>On the display tab click 'Open stroke display'. With Plover disabled you should be able to hit keys on your keyboard and see them show up in the stroke display window. Use this to make sure you have set up your keymap correctly. You are now ready to steno!</p>
<h2 id="page_Learning-Stenography">Learning Stenography</h2>
<ul>
<li>
<a href="https://sites.google.com/site/ploverdoc/" class="Link--external">Learn Plover!</a>
</li>
<li>
<a href="http://qwertysteno.com/Home/" class="Link--external">QWERTY Steno</a>
</li>
<li>
<a href="https://joshuagrams.github.io/steno-jig/" class="Link--external">Steno Jig</a>
</li>
<li>More resources at the Plover <a href="https://github.com/openstenoproject/plover/wiki/Learning-Stenography" class="Link--external">Learning Stenography</a> wiki</li>
</ul>
<h2 id="page_Interfacing-with-the-code">Interfacing with the code</h2>
<p>The steno code has three interceptible hooks. If you define these functions, they will be called at certain points in processing; if they return true, processing continues, otherwise it's assumed you handled things.</p>
<pre><code class="language-C">bool send_steno_chord_user(steno_mode_t mode, uint8_t chord[6]);
</code></pre>
<p>This function is called when a chord is about to be sent. Mode will be one of <code>STENO_MODE_BOLT</code> or <code>STENO_MODE_GEMINI</code>. This represents the actual chord that would be sent via whichever protocol. You can modify the chord provided to alter what gets sent. Remember to return true if you want the regular sending process to happen.</p>
<pre><code class="language-C">bool process_steno_user(uint16_t keycode, keyrecord_t *record) { return true; }
</code></pre>
<p>This function is called when a keypress has come in, before it is processed. The keycode should be one of <code>QK_STENO_BOLT</code>, <code>QK_STENO_GEMINI</code>, or one of the <code>STN_*</code> key values.</p>
<pre><code class="language-C">bool postprocess_steno_user(uint16_t keycode, keyrecord_t *record, steno_mode_t mode, uint8_t chord[6], int8_t pressed);
</code></pre>
<p>This function is called after a key has been processed, but before any decision about whether or not to send a chord. If <code>IS_PRESSED(record-&gt;event)</code> is false, and <code>pressed</code> is 0 or 1, the chord will be sent shortly, but has not yet been sent. This is where to put hooks for things like, say, live displays of steno chords or keys.</p>
<h2 id="page_Keycode-Reference">Keycode Reference</h2>
<p>As defined in <code>keymap_steno.h</code>.</p>
<blockquote>
<p>Note: TX Bolt does not support the full set of keys. The TX Bolt implementation in QMK will map the GeminiPR keys to the nearest TX Bolt key so that one key map will work for both.</p>
</blockquote>
<table>
<thead>
<tr>
<th>GeminiPR</th>
<th>TX Bolt</th>
<th>Steno Key</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>STN_N1</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #1</td>
</tr>
<tr>
<td><code>STN_N2</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #2</td>
</tr>
<tr>
<td><code>STN_N3</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #3</td>
</tr>
<tr>
<td><code>STN_N4</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #4</td>
</tr>
<tr>
<td><code>STN_N5</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #5</td>
</tr>
<tr>
<td><code>STN_N6</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #6</td>
</tr>
<tr>
<td><code>STN_N7</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #7</td>
</tr>
<tr>
<td><code>STN_N8</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #8</td>
</tr>
<tr>
<td><code>STN_N9</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #9</td>
</tr>
<tr>
<td><code>STN_NA</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #A</td>
</tr>
<tr>
<td><code>STN_NB</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #B</td>
</tr>
<tr>
<td><code>STN_NC</code></td>
<td><code>STN_NUM</code></td>
<td>Number bar #C</td>
</tr>
<tr>
<td><code>STN_S1</code></td>
<td><code>STN_SL</code></td>
<td><code>S-</code> upper</td>
</tr>
<tr>
<td><code>STN_S2</code></td>
<td><code>STN_SL</code></td>
<td><code>S-</code> lower</td>
</tr>
<tr>
<td><code>STN_TL</code></td>
<td><code>STN_TL</code></td>
<td><code>T-</code></td>
</tr>
<tr>
<td><code>STN_KL</code></td>
<td><code>STN_KL</code></td>
<td><code>K-</code></td>
</tr>
<tr>
<td><code>STN_PL</code></td>
<td><code>STN_PL</code></td>
<td><code>P-</code></td>
</tr>
<tr>
<td><code>STN_WL</code></td>
<td><code>STN_WL</code></td>
<td><code>W-</code></td>
</tr>
<tr>
<td><code>STN_HL</code></td>
<td><code>STN_HL</code></td>
<td><code>H-</code></td>
</tr>
<tr>
<td><code>STN_RL</code></td>
<td><code>STN_RL</code></td>
<td><code>R-</code></td>
</tr>
<tr>
<td><code>STN_A</code></td>
<td><code>STN_A</code></td>
<td><code>A</code> vowel</td>
</tr>
<tr>
<td><code>STN_O</code></td>
<td><code>STN_O</code></td>
<td><code>O</code> vowel</td>
</tr>
<tr>
<td><code>STN_ST1</code></td>
<td><code>STN_STR</code></td>
<td><code>*</code> upper-left</td>
</tr>
<tr>
<td><code>STN_ST2</code></td>
<td><code>STN_STR</code></td>
<td><code>*</code> lower-left</td>
</tr>
<tr>
<td><code>STN_ST3</code></td>
<td><code>STN_STR</code></td>
<td><code>*</code> upper-right</td>
</tr>
<tr>
<td><code>STN_ST4</code></td>
<td><code>STN_STR</code></td>
<td><code>*</code> lower-right</td>
</tr>
<tr>
<td><code>STN_E</code></td>
<td><code>STN_E</code></td>
<td><code>E</code> vowel</td>
</tr>
<tr>
<td><code>STN_U</code></td>
<td><code>STN_U</code></td>
<td><code>U</code> vowel</td>
</tr>
<tr>
<td><code>STN_FR</code></td>
<td><code>STN_FR</code></td>
<td><code>-F</code></td>
</tr>
<tr>
<td><code>STN_PR</code></td>
<td><code>STN_PR</code></td>
<td><code>-P</code></td>
</tr>
<tr>
<td><code>STN_RR</code></td>
<td><code>STN_RR</code></td>
<td><code>-R</code></td>
</tr>
<tr>
<td><code>STN_BR</code></td>
<td><code>STN_BR</code></td>
<td><code>-B</code></td>
</tr>
<tr>
<td><code>STN_LR</code></td>
<td><code>STN_LR</code></td>
<td><code>-L</code></td>
</tr>
<tr>
<td><code>STN_GR</code></td>
<td><code>STN_GR</code></td>
<td><code>-G</code></td>
</tr>
<tr>
<td><code>STN_TR</code></td>
<td><code>STN_TR</code></td>
<td><code>-T</code></td>
</tr>
<tr>
<td><code>STN_SR</code></td>
<td><code>STN_SR</code></td>
<td><code>-S</code></td>
</tr>
<tr>
<td><code>STN_DR</code></td>
<td><code>STN_DR</code></td>
<td><code>-D</code></td>
</tr>
<tr>
<td><code>STN_ZR</code></td>
<td><code>STN_ZR</code></td>
<td><code>-Z</code></td>
</tr>
<tr>
<td><code>STN_FN</code></td>
<td>(GeminiPR only)</td>
<td></td>
</tr>
<tr>
<td><code>STN_RES1</code></td>
<td>(GeminiPR only)</td>
<td></td>
</tr>
<tr>
<td><code>STN_RES2</code></td>
<td>(GeminiPR only)</td>
<td></td>
</tr>
<tr>
<td><code>STN_PWR</code></td>
<td>(GeminiPR only)</td>
<td></td>
</tr>
</tbody>
</table>
</div>
<nav>
<ul class="Pager">
<li class=Pager--prev><a href="../Features/Space_Cadet_Shift_Enter.html">Previous</a></li> <li class=Pager--next><a href="../Features/Swap_Hands.html">Next</a></li> </ul>
</nav>
</article>
</div>
</div>
</div>
</div>
<!-- JS -->
<script src="../themes/daux/js/jquery-1.11.3.min.js"></script><script src="../themes/daux/js/highlight.pack.js"></script><script src="../themes/daux/js/daux.js"></script>
<!-- Tipue Search -->
<script type="text/javascript" src="../tipuesearch/tipuesearch.js"></script>
<script>
window.onunload = function(){}; // force $(document).ready to be called on back/forward navigation in firefox
$(function() {
tipuesearch({
'base_url': '../'
});
});
</script>
</body>
</html>