TabNav
Sublime Text plugin for keyboard navigation of tables in text files. Supports: Markdown, Org Mode, Textile, CSV.
Details
Installs
- Total 490
- Win 258
- Mac 150
- Linux 82
Nov 21 | Nov 20 | Nov 19 | Nov 18 | Nov 17 | Nov 16 | Nov 15 | Nov 14 | Nov 13 | Nov 12 | Nov 11 | Nov 10 | Nov 9 | Nov 8 | Nov 7 | Nov 6 | Nov 5 | Nov 4 | Nov 3 | Nov 2 | Nov 1 | Oct 31 | Oct 30 | Oct 29 | Oct 28 | Oct 27 | Oct 26 | Oct 25 | Oct 24 | Oct 23 | Oct 22 | Oct 21 | Oct 20 | Oct 19 | Oct 18 | Oct 17 | Oct 16 | Oct 15 | Oct 14 | Oct 13 | Oct 12 | Oct 11 | Oct 10 | Oct 9 | Oct 8 | |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
Windows | 1 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 1 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
Mac | 0 | 0 | 0 | 1 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 1 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
Linux | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
Readme
- Source
- raw.githubusercontent.com
TabNav - Keyboard Navigation of Tabular Data
TabNav is a Sublime Text plugin for keyboard navigation of tabular text data. Quickly move and select “cells” of text in the following formats, without taking your hands off the keyboard:
- Markdown pipe tables
- Org Mode tables
- Textile tables
- CSV files
TabNav also provides the ability to copy only the contents of the table, excluding markup, in a format that can be readily-pasted into other programs, such as Excel.
Table of Contents
- Installation Instructions
- Recommended Key Bindings
- Commands
- Contexts
- Capture Levels
- Key Bindings
- Customization
Installation Instructions
Package Control
- If you haven't already, install Package Control.
- In the Sublime Text command palette, run the command:
Package Control: Install Package
. - Select
TabNav
from the list of available packages.
Git Clone
- Clone this repository to your local machine:
git clone https://github.com/mitchvm/tabnav.git
- If you didn't clone directly to the Sublime Text packages directory, create a symbolic link to the repository in your local Sublime Text packages directory.
Manual
- Download the latest TabNav release and unzip it to your local packages directory.
- To find the local packages directory, open the Sublime Text Preferences and select “Browse packages…”
- Note: All of the TabNav files should be directly under a
tabnav
directory in thePackages
directory. If the files are nested further, Sublime Text will not find them.
Recommended Key Bindings
:warning: TabNav has no keybindings enabled on initial install.
A package like TabNav will obviously require many key bindings. In an effort to not clobber either the default Sublime Text key bindings, or the key bindings of other packages you may have installed, while also allowing maximum flexibility for configuring key bindings based on your own personal preferences and keyboard layout, no key bindings are configured by default.
A set of recommended key bindings is provided in the package's key bindings files, however they are all commented out. The recommended key bindings are based on a US-English QWERTY keyboard. They make heavy use of the cluster of four keys immediately to the left of the Enter key.
Key binding setup
- From the Sublime Text Main menu, select Preferences ❯ Package Settings ❯ TabNav ❯ Key Bindings
- This will open the the TabNav key bindings package key bindings file (on the left) along with your user key bindings file (on the right) in a new window. Notice that all of the default key bindings have been commented out with
//
at the start of each line.
- This will open the the TabNav key bindings package key bindings file (on the left) along with your user key bindings file (on the right) in a new window. Notice that all of the default key bindings have been commented out with
- Copy commented-out key TabNav bindings to into your user key bindings array.
- You must paste the key bindings inside of the outer-most array brackets in your user key bindings file.
With the copied key bindings still selected (and still commented out), un-comment the entire selection. (Main menu: Edit ❯ Comment ❯ Toggle Comment)
- If you have no other custom key bindings, your user key bindings file should look like this. Notice the brackets -
[
and]
- on the first and last lines, respectively, and no//
at the start of each line other than comment lines.
[ // =================== TabNav Key Bindings ========================= // #### TabNav: Non-navigation keybindings #### { // Enable TabNav on current view // Note: this keybinding gets clobbered by the "select cell on right" keybindings // once TabNav is enabled. In a CSV file, what this means is that the first press // of this keybinding enables TabNav, and the next press selects the current cell. "keys": ["ctrl+'"], "command": "enable_tabnav" }, ... (a lot more key bindings here) // =================== End: TabNav Key Bindings ===================== ]
- If you have no other custom key bindings, your user key bindings file should look like this. Notice the brackets -
See the Key Bindings section for more details on the recommended key bindings, as well as how to use custom key bindings.
Commands
TabNav adds the following commands to Sublime Text. They are all accessible via the TabNav submenu under the Selection menu.
Note: If you're reading this on packagecontrol.io, the tables render more clearly when read on GitHub.
Table Navigation Commands
The table navigation commands below only operate within the context of a table. All of the table commands are compatible with multiple cursors, and even multiple cursors in multiple, disjoint tables.
As noted above, TabNav has no key bindings enabled by default. The key bindings shown below are the recommended bindings. The core movement and selection key bindings combine one of four basic modifier key combinations together with with one of the four direction keys:
Name | Windows/Linux | macOS | Description |
---|---|---|---|
Move cursor to cell… | Alt | Moves all cursors to the next cell in the desired direction. | |
Select next cell… | Ctrl | ⌘ | Moves all selections to the adjacent cell in the desired direction. |
Select last cell… | CtrlAlt | ⌘ | Moves all selections to the furthest cell in the row/column in the desired direction. |
Extend selection… | CtrlShift | ⌘⇧ | Adds the next cell in the desired direction to the current selections. |
Extend selection to end of row/column… | CtrlAltShift | ⌘⇧ | Selects all cells in the row/column between the currently selected cell and the end of the row/column in the desired direction. |
Reduce selection… | AltShift | ⇧ | When two or more cells in sequence are selected, removes the selection from a cell in the desired direction. |
Add cursor to cell… | For each active cursor, add an additional cursor to the cell in the desired direction. The recommended key bindings do not include these commands. | ||
Remove cursor from cell… | When two or more cells in sequences contain cursors, removes the cursors from a cell in the desired direction. The recommended key bindings do not include these commands. |
Direction Key |
---|
Up [ |
Left ; | ' Right |
/ Down |
Beyond the core navigation commands, these additional movement and selection commands are provided. Unlike most of the core commands, all of these commands are idempotent - that is, they generate the same Sublime Text selections/cursors regardless of how many times they are invoked, even if the current selections/cursors are already aligned with table cells. This might prove useful, for example, if recording a macro.
Name | Windows/Linux Key binding | macOS Key binding |
---|---|---|
Move cursor to start of current cell1 | ||
Move cursor to end of current cell1 | ||
Select text to start of current cell | ||
Select text to end of current cell | ||
Select current cell2 | ||
Select row cells | CtrlShift+L | ⌘⇧+L |
Select column cells | CtrlShift+C | ⌘⇧+C |
Select all table cells | CtrlAltShift+C | ⌘⇧+C |
1 On initial invocation, the core move cursor left/right commands will also move the cursor to the start/end of the current cell, respectively, if not all selections are already at that position.
2 On initial invocation, the core select next and extend selection commands will also select the current cell, if not all existing selections line up with table cells.
Other Commands
These commands will operate even outside the context of a table. They are accessible via the Command Palette.
Name | Windows/Linux Key Binding | macOS Key Binding | Description |
---|---|---|---|
Enable on current view | Ctrl+' | ⌘+' | Enables TabNav on the current view. Note, once enabled, the key binding is clobbered by the “Move cursor to cell on right” command (if using the recommended key bindings). |
Disable on current view | Disables TabNav on the current view. | ||
Set capture level | Configures the selection capture level in use on the current view. | ||
Reset capture level to previous | Resets to the previously configured capture level. Useful for recording macros (i.e., change capture level, perform action, reset capture level). | ||
Set CSV delimiter | Sets the delimiter to use for CSV files. See the CSV context section for more information. | ||
Trim whitespace from selections | Removes all whitespace characters from either end of all current selections. | ||
Merge adjacent selections | Merges selected regions that share a common start/end point. Useful if wanting to cut/paste multiple adjacent columns. Use with the cell capture level. |
||
Copy selections as TSV | Copies all current selections as tab-delimited data, with all selections on the same row of text tab-separated and a newline between selection row. This is useful, for example, to copy data from a text table into Excel. | ||
Copy selections with delimiter | Same as the “Copy selections as TSV” command, but prompts the user to input the delimiter to use. |
Contexts
TabNav operates on the concept of “contexts”, which define how it identifies tabular data in a document. By default, it includes context definitions for Markdown, Org Mode, Textile, and CSV documents.
Markdown
TabNav is enabled by default in Markdown documents. Only “pipe” style tables are supported. Other styles of Markdown tables are not currently supported.
Some flavours of Markdown support “borderless” tables, where pipes are not required on the outer edges of the table. For example, this is a valid table:
| Heading 1 | Heading 2 | Heading 3 |
|:----------|:----------|----------:|
| 1.1 | 1.2 | 1.3 |
| 2.1 | 2.2 | 2.3 |
Alternatively, the same table as a “bordered” table would look like this:
| Heading 1 | Heading 2 | Heading 3 |
|:----------|:----------|----------:|
| 1.1 | 1.2 | 1.3 |
| 2.1 | 2.2 | 2.3 |
By default, TabNav supports both borderless and bordered tables. To be able to support borderless tables, however, any line of (non-raw) text containing a pipe character is considered to be part of a table.
Since this is a Markdown file, if I put a pipe right here | then this line of text is considered to be a table with two cells.
If you only use bordered Markdown tables, you can configure TabNav to be more restrictive in what it considers to be a table. See Context Configuration.
Org Mode
TabNav is enabled by default in Org Mode documents, using the scopes defined by either the orgmode or orgextended packages. Tables in “raw” scopes are ignored.
TabNav recognizes three kinds of markup rows in Org Mode tables:
Textile
TabNav is enabled by default in Textile documents. Textile tables differ from most of the other supported markup languages in two ways:
- Cells and rows may contain inline table markup in addition to content.
- Markup rows might not contain “cells” of markup that are aligned with the table cells.
Inline markup directly in the cell respects the current capture level - when set to trimmed
or content
, the markup is omitted from selections; when set to markup
or cell
, the markup is included in selections. However, row markup - that is, styles and classes assigned to the row, before the first pipe - and stand-alone markup rows without a pipe at the end of the row are always ignored, regardless of capture level. Header and footer rows are treated as normal content rows.
Textile also supports cells spanning multiple rows and/or columns, though TabNav makes no special effort to support row and column spanning. When moving a single cursor/selection, the behaviour is mostly how you would expect, or at least predictable, but when it comes to selecting regions of the table, cells spanning multiple rows or columns tend to give pretty funky results.
At this time, there is no intention of adding “proper” support for cells spanning multiple rows or columns in Textile tables.
CSV
CSV requires special handling, specifically because there are so many permutations of “separated value” documents. There is no specific Sublime Text scope for CSV documents. Rather, TabNav treats CSV as the fall-back context if no other context was positively identified, though TabNav is disabled by default in CSV contexts - use the “Enable on current view” command to enable it.
TabNav integrates with both the Advanced CSV and Rainbow CSV3 packages. If the syntax on the current view comes from either of those packages, the delimiter being used by them is also automatically used by TabNav.
If the syntaxes provided by those two packages are not in use on the current view, then TabNav attempts to infer the delimiter to use by inspecting the first line of the file. If the first line of the file contains only one of the following characters, then that character is assumed to be the delimiter:
- Comma:
,
- Semi-colon:
;
- Pipe:
|
- Tab
You can also specify a particular delimiter to use (when not in an Advanced CSV or Rainbow CSV syntax) with the “Set CSV delimiter” command. Note that a space cannot be used as the delimiter for the built-in CSV context.
Finally, if all other methods of determining the delimiter fail, TabNav uses a comma as the default delimiter.
3 TabNav only partially supports Rainbow CSV syntaxes. The two restrictions are: only single-character delimiters are supported, and all CSV files are treated as quoted files, regardless if using a Rainbow CSV “simple” syntax.
Capture Levels
TabNav contexts provide multiple “capture levels” that define how much text to select within a table cell. The available capture levels are listed below. Each level captures all of the text as the level above, and potentially more.
- Trimmed: Only the text contained within the cell, excluding any whitespace on either side. Cells containing only markup are omitted from selections.
- Content: The text in the cell as well as any whitespace around the text. Cells containing only markup are omitted from selections.
- Markup: The content of the cell, plus any markup in the cell, but excluding the delimiter. All table cells, including those without any markup, are included in selections.
- Cell: The entirety of all table cells are included in selections, including the delimiter preceding each cell, if applicable.
Note that not all capture levels are relevant to all contexts - CSV, for example, does not contain any markup beyond the cell delimiter. What's more, most contexts do not mix markup and content within the same cell.
The default capture level is content
. The capture level in use on a particular view can be changed with the “Set capture level” command. The default capture level can be configured globally or per-context.
Key Bindings
Simply due to the nature of the package, TabNav requires many key bindings. The recommended key bindings include several that override built-in Sublime Text key bindings. Effort has been made to minimize the impact on the default Sublime Text key bindings by having the TabNav key bindings take effect only under very specific circumstances. Only if all of the following conditions are met will the TabNave key bindings override the built-in key bindings.
- There is a TabNav context configured that matches the current view.
- TabNav is enabled on the current view - in most contexts it is enabled by default.
- The start point of the first selection is within a table.
The following built-in Sublime Text key bindings get overridden by the recommended TabNav key bindings:
Operating System | Key Binding | Sublime Text Command | TabNav Command |
---|---|---|---|
Windows, Linux | Ctrl+; | Open “go to word” overlay | Select cell(s) left |
Windows, Linux | Ctrl+[ | Un-indent line(s) | Select cell(s) up |
Windows, Linux | Ctrl+/ | Comment line(s) | Select cell(s) down |
Windows, Linux | CtrlShift+[ | Fold selection(s) | Extend selection(s) up |
Windows, Linux | CtrlShift+/ | Insert comment | Extend selection(s) down |
Windows, Linux | CtrlShift+L | Split selection(s) into lines | Select cells in table row(s) |
macOS | ⌘+[ | Un-indent line(s) | Select cell(s) up |
macOS | ⌘⇧+[ | Fold selection(s) | Extend selection(s) up |
macOS | ⌘⇧+L | Split selection(s) into lines | Select cells in table row(s) |
To temporarily disable most of the recommended TabNav keybindings, it is enough to disable TabNav on the current view. The only (recommended) key binding that is not disabled by doing this is the binding to enable TabNav on the view.
Custom Key Bindings
Of course, it is entirely valid to ignore the recommended key bindings and use your own custom key bindings, or add more key bindings for commands not covered by the recommended bindings.
All of the available commands are and arguments enumerated in the CommandListing
file.
For each navigation key binding, it is recommended to add the is_tabnav_context
key binding context to limit the scopes within which the key binding will take effect, as described above.
Customization
TabNav offers considerable configuration, or even customizability to modify the default contexts' behaviour or add new contexts.
Configuration Options
Selecting the Preferences ❯ Package Settings ❯ TabNav ❯ Settings - TabNav menu item opens the TabNav default settings file, as well as your local TabNav settings file. Override the default configurations by placing the parameter into your local settings file. The following global configuration parameters are available:
capture_level
: The initial capture level to use. The capture level can also be configured per-context, or changed on the active view using the “Set capture level” command. Options:trimmed
,content
,markup
,cell
. Default:content
.trim_on_copy
: When true, the “Copy selections” commands trim whitespace from the selected regions' text prior to putting it on the clipboard. The selections in the view themselves are not altered. Default:true
.enable_explicitly
: When false, TabNav is assumed to be enabled if a context is successfully matched to the file. When true, TabNav must be explicitly enabled on each view. This setting can also be configured per-context. Default:false
.log_level
: Set toINFO
orDEBUG
to see TabNav log messages in the Sublime Text console. DefaultWARNING
.
Context Configuration
To modify the behaviour of the default contexts, or to add new contexts, use the user_contexts
4 element in your local TabNav settings file
To override a default context's setting, you only need to provide the path to that setting in the user_contexts
element; you don't need to copy the full context definition. For example, to configure the Markdown context to only support bordered tables, add this to your user configuration:
{
"user_contexts":
{
"markdown":
{
"patterns": [
{
"line": "^(?P<table>(\\|\\s*[:-]+\\s*(?=\\|))+\\|)$",
"cell": "(?P<cell>\\|(?P<markup>\\s*[:-]+\\s*))(?=\\|)"
},
{
"line": "^(?P<table>\\|.*\\|)$",
"cell": "(?P<cell>\\|(?P<content>\\s*(?P<trimmed>.*?)\\s*))(?=\\|)"
}
]
}
}
}
See the Custom Contexts section below for descriptions of the standard context parameters.
4 The default settings file has a contexts
element. TabNav merges settings from the user_contexts
and contexts
settings. If you want to completely overwrite the default contexts, you can use the contexts
element in your local settings file, however this is not recommended.
CSV Context Configuration
The auto_csv
context is a special case with several custom parameters in addition to the standard context parameters.
auto_delimiters
: The list of delimiters that TabNav will check when attempting to infer the CSV delimiter from the first line of the file.default_delimiter
: The ultimate fallback delimiter used for the CSV context if all other methods of determining the delimiter fail.
Custom Contexts
Additional contexts can also be defined in the user_contexts
element. For examples, see the tabnav.sublime-settings
file (Preferences ❯ Package Settings ❯ TabNav ❯ Settings - TabNav) to see the default contexts, which are generously commented.
The following parameters are used to define a TabNav context:
selector
: Required. A Sublime Text selector that identifies the scope within which the context operates. If multiple selections are currently active, only the first selection's scope is checked. If multiple TabNav contexts' selectors match the current scope, then the context with the highest selector “score” (as returned by the Sublime Text API) is used.except_selector
: Optional. A Sublime Text selector that overrides the baseselector
. If the first selection matches this selector, then the context is not matched.patterns
: Required. One or more pattern definitions used to identify and parse rows of table content. If only one pattern is provided, it need-not be placed in a JSON array. If multiple patterns are provided, they are applied in sequence until the first match. In general, patterns for markup rows should be placed above patterns for content rows.enable_explicitly
: Optional. A boolean to indicate if the TabNav must be explicitly enabled when this context is matched. Overrides the globalenable_explicitly
setting. Defaultfalse
.capture_level
: Optional. The default capture level to use with this context. Overrides the globalcapture_level
setting. Possible values:trimmed
,content
,markup
,cell
.
Pattern Definitions
The each element of the patterns
context parameter defines how to:
- identify that a line of text is part of a table, and
- parse the contents of the table cells of that line of text.
Each pattern contains two elements, explained in more detail below.
line
: Optional. A single regular expression that is used to determine if the line of text is part of a table.cell
: Required. One or more regular expressions that are used to identify cell contents from a line of text.
line
capture group
The optional line
expression is used to determine if the pattern applies to the line of text and, if so, the portion of the line of text that constitutes the table. If used, the line
expression must return a named table
group that captures the part of the line to use for matching table content with the cell
expressions. If a line
expression is not provided with the pattern, the entire line of text is parsed using the cell
expressions.
There is one scenario where the line
element is required, and the cell
element can be omitted: to capture a line of a table, but not parse any cells from the line, include a line
element without a table
capture group.
cell
capture groups
Each element of the cell
expressions should contain up to four nested named capture groups:
(cell (markup (content (trimmed))))
These correspond to the four TabNav capture levels. All outer capture groups must be defined if an inner group is to be used, regardless if the outer group captures anything additional to the inner group or not. For example, most table formats do not contain markup within content cells, however, the markup
group must be included to be able to capture the content
group.
To define markup-only cells that should not be included in selections at the content
or trimmed
levels, omit the content
and trimmed
capture groups.
The cell
capture group should capture the delimiter that precedes the content of the cell, if applicable. In “borderless” contexts, such as CSV and borderless Markdown tables, the first cell of the line does not capture a delimiter.
Each match of each expression in the array should return a single cell's contents. Each expression can also, optionally, return a zero-width match immediately prior to the last matching delimiter. This match will be ignored.
If multiple expressions are provided in a JSON array, they are each processed in sequence until their matches are exhausted. Use this to have, for example, one expression to capture the first cell of the row, a different expression to capture cells in the middle of the row, and a third expression to capture the final cell. If only one expression is provided, it need-not be placed in a JSON array.
Example
One table format that mixes markup and content in a table cell is Textile. Here is a sample row of a Textile table that defines headers. Cells are pipe-delimited, and the _.
at the start of the cell is markup indicating that the cell is a header cell.
|_. First Header |_. Second Header |
Here is a visual representation of what the four TabNav capture groups should capture from this row:
|_. First Header |_. Second Header |
↑↑ ↑↑ ↑ ↑↑↑ ↑↑ ↑↑
|| |└ trimmed ─┘ ||| |└ trimmed ──┘|
|| └─ content ───┤|| └─ content ───┤
|└─── markup ────┤|└─── markup ────┤
└──── cell ──────┘└──── cell ──────┘
Alternatively, presented as a table (how meta):
Capture Group | First Cell Selection | Second Cell Selection |
---|---|---|
cell |
|_. First Header |
|_. Second Header |
markup |
_. First Header |
_. Second Header |
content |
First Header |
Second Header |
trimmed |
First Header |
Second Header |
Notice that the final |
is not captured as part of any cell - each cell only captures the preceding delimiter.