diff options
author | Preston Pan <preston@nullring.xyz> | 2023-06-11 17:43:34 -0700 |
---|---|---|
committer | Preston Pan <preston@nullring.xyz> | 2023-06-11 17:43:34 -0700 |
commit | 846b48b0b179fcaa49c494d0da0e23db71b989dd (patch) | |
tree | ad76990055afed803a8de5ed08b89a73e8a32967 |
first commit
-rw-r--r-- | .gitignore | 1 | ||||
-rw-r--r-- | LICENSE.org | 14 | ||||
-rw-r--r-- | README.org | 25 | ||||
-rw-r--r-- | config/doom.org | 174 | ||||
-rw-r--r-- | config/elfeed.org | 20 | ||||
-rw-r--r-- | config/fish.org | 55 | ||||
-rw-r--r-- | config/index.org | 27 | ||||
-rw-r--r-- | config/kmonad.org | 720 | ||||
-rw-r--r-- | config/qtile.org | 286 | ||||
-rw-r--r-- | config/qutebrowser.org | 99 | ||||
-rw-r--r-- | index.org | 37 | ||||
-rw-r--r-- | journal/20230609.org | 10 | ||||
-rw-r--r-- | journal/20230610.org | 18 | ||||
-rw-r--r-- | journal/20230611.org | 16 | ||||
-rw-r--r-- | journal/index.org | 28 | ||||
-rw-r--r-- | mindmap/duality.org | 9 | ||||
-rw-r--r-- | mindmap/emergence.org | 9 | ||||
-rw-r--r-- | mindmap/everything.org | 36 | ||||
-rw-r--r-- | mindmap/factorial.org | 8 | ||||
-rw-r--r-- | mindmap/index.org | 21 | ||||
-rw-r--r-- | mindmap/python.org | 6 | ||||
-rw-r--r-- | mindmap/recursion.org | 71 | ||||
-rw-r--r-- | mindmap/self-assembly.org | 9 | ||||
-rw-r--r-- | mindmap/stack.org | 6 | ||||
-rw-r--r-- | style.css | 323 | ||||
-rwxr-xr-x | track.sh | 8 |
26 files changed, 2036 insertions, 0 deletions
diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..849e6ba --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +./website/** diff --git a/LICENSE.org b/LICENSE.org new file mode 100644 index 0000000..df18b21 --- /dev/null +++ b/LICENSE.org @@ -0,0 +1,14 @@ +#+title: The MIT License +#+author: Preston Pan +#+description: license for website +#+html_head: <link rel="stylesheet" type="text/css" href="./style.css" /> +#+date: <2023-06-11 Sun> +#+language: en + +Copyright © 2023 Preston Pan + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/README.org b/README.org new file mode 100644 index 0000000..f2c734d --- /dev/null +++ b/README.org @@ -0,0 +1,25 @@ +#+title: ret2pop +#+author: Preston Pan +#+date: <2023-06-11 Sun> +#+description: My website for all things. +#+options: num:nil broken-links:t tex:dvipng +#+html_head: <link rel="stylesheet" type="text/css" href="style.css" /> + +* What is this? +It's ret2pop! My website for all things math. Wait, no. Computer science. +Music. Okay, let me try that again. + +It's ret2pop! My website for all things. + +** Sounds awesome! But why is it written in org instead of html? +Because I have converted to the church of emacs. Remember: +anything and everything that can be rewritten in org, can and should +be rewritten in org. +*** so how did you actually do that? +By using the ~org-publish~ function, ~org-roam~, and ~org-journal~, +as well as writing a lot of things with a literate config. +** So, where is this website of yours located? +https://ret2pop.nullring.xyz, it is the road to enlightenment. + +* License +See the [[file:LICENSE.org][license]]. The style.css has its own license. diff --git a/config/doom.org b/config/doom.org new file mode 100644 index 0000000..b3fce4b --- /dev/null +++ b/config/doom.org @@ -0,0 +1,174 @@ +#+title: Doom Literate Config +#+author: Preston Pan +#+date: <2023-06-09 Fri> +#+description: My doom emacs configuration +#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" /> + +* Configuration +** Basic Information +My name, and the org mode directory on my computer, as well as basic editor configuration options. +Below is the old documentation. +#+begin_src emacs-lisp :tangle yes +(setq user-full-name "Preston Pan" + user-mail-address "prseton@nullring.xyz") + +(setq display-line-numbers-type t) + +(setq org-directory "~/org/") +#+end_src +- `load!' for loading external *.el files relative to this one +- `add-load-path!' for adding directories to the `load-path', relative to + this file. Emacs searches the `load-path' when you load packages with + `require' or `use-package'. +** Font +Now we configure fonts: +#+begin_src emacs-lisp :tangle yes +(setq doom-font (font-spec :family "FiraCode Nerd Font" :size 14 :weight 'semi-light) + doom-variable-pitch-font (font-spec :family "Fira Sans" :size 14) + doom-unicode-font (font-spec :family "Symbola" :size 14) + doom-serif-font (font-spec :family "Fira Sans" :size 14) + doom-big-font (font-spec :family "FiraCode Nerd Font" :size 28)) +#+end_src +** Color Scheme +I'm using the catppuccin theme, which is available on github. +#+begin_src emacs-lisp :tangle yes +(setq doom-theme 'catppuccin) +(setq catppuccin-flavor 'mocha) +#+end_src +** Doom Module and Programs Configuration +*** IRC +Set up circe to connect to my bouncer: +#+begin_src emacs-lisp :tangle yes +(after! circe + (set-irc-server! "nullring.xyz" + `(:tls t + :port 4095 + :nick "LiCoO2/AndreiNet" + :user "LiCoO2/AndreiNet" + :pass ,(+pass-get-secret "ZNC")))) +#+end_src +*** Email +In order to use this configuration, you must install and configure mu and mbsync. +#+begin_src emacs-lisp :tangle yes +(setq mu4e-get-mail-command "mbsync --config ~/.config/doom/mbsyncrc prestonpan") + +(setq send-mail-function 'smtpmail-send-it) +(setq smtpmail-smtp-server "mail.nullring.xyz") +(setq smtpmail-smtp-service 465) +(setq smtpmail-stream-type 'ssl) + +(setq mu4e-drafts-folder "/Drafts") +(setq mu4e-sent-folder "/Sent") +(setq mu4e-trash-folder "/Trash") + +(setq mu4e-html2text-command "w3m -T text/html" + mu4e-update-interval 300 + mu4e-headers-auto-update t + mu4e-view-show-images t + mu4e-compose-signature-auto-include nil + mu4e-use-fancy-chars t) +(setq mu4e-compose-reply-ignore-address '("no-?reply" "preston@nullring.xyz")) +#+end_src +*** RSS +We need to set up elfeed with a list of rss feeds. +#+begin_src emacs-lisp :tangle yes +(after! elfeed + (setq elfeed-search-filter "@1-month-ago +unread")) +(add-hook! 'elfeed-search-mode-hook #'elfeed-update) +(setq rmh-elfeed-org-files '("~/org/elfeed.org")) +#+end_src +*** Music +In order to use this configuration, you must have mpd configured to use the same directory. +We automatically connect to mpd. +#+begin_src emacs-lisp :tangle yes +(setq emms-player-mpd-music-directory "~/music/") +(setq emms-player-list '(emms-player-mpd)) +(emms-player-mpd-connect) +#+end_src +*** Journal +First we set the journal to be in the website directory: +#+begin_src emacs-lisp :tangle yes + +(setq org-journal-dir "~/org/website/journal/") +(setq org-journal-date-format "%A, %d %B %Y") +#+end_src +And then we add the headers needed to export the journal automatically: +#+begin_src emacs-lisp :tangle yes +(defun org-journal-file-header-func (time) + "Custom function to create journal header." + (concat + (pcase org-journal-file-type + (`daily "#+TITLE: Daily Journal\n#+STARTUP: showeverything\n#+DESCRIPTION: My daily journal entry\n#+AUTHOR: Preston Pan\n#+HTML_HEAD: <link rel=\"stylesheet\" type=\"text/css\" href=\"../style.css\" />\n#+html_head: <script src=\"https://polyfill.io/v3/polyfill.min.js?features=es6\"></script>\n#+html_head: <script id=\"MathJax-script\" async src=\"https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-mml-chtml.js\"></script>") + (`weekly "#+TITLE: Weekly Journal\n#+STARTUP: folded") + (`monthly "#+TITLE: Monthly Journal\n#+STARTUP: folded") + (`yearly "#+TITLE: Yearly Journal\n#+STARTUP: folded")))) + +(setq org-journal-file-header 'org-journal-file-header-func) +(setq org-journal-file-format "%Y%m%d.org") +#+end_src +*** Brain +#+begin_src emacs-lisp :tangle yes +(setq org-brain-path "~/org/website/brain/") +#+end_src +*** Roam +#+begin_src emacs-lisp :tangle yes +(setq org-roam-directory (file-truename "~/org/website/mindmap")) +(setq org-roam-capture-templates '(("d" "default" plain "%?" + :target (file+head "${title}.org" + "#+title: ${title}\n#+author: Preston Pan\n#+html_head: <link rel=\"stylesheet\" type=\"text/css\" href=\"../style.css\" />\n#+html_head: <script src=\"https://polyfill.io/v3/polyfill.min.js?features=es6\"></script>\n#+html_head: <script id=\"MathJax-script\" async src=\"https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-mml-chtml.js\"></script>") + :unnarrowed t))) +#+end_src +*** Publishing +#+begin_src emacs-lisp :tangle yes +(require 'ox-publish) +(setq org-publish-project-alist + '(("website-org" + :base-directory "~/org/website" + :base-extension "org" + :publishing-directory "~/website_html" + :recursive t + :publishing-function org-html-publish-to-html + :headline-levels 4 + :auto-preamble t) + ("website-static" + :base-directory "~/org/website" + :base-extension "css\\|js\\|png\\|jpg\\|gif\\|pdf\\|mp3\\|ogg\\|swf" + :publishing-directory "~/website_html/" + :recursive t + :publishing-function org-publish-attachment) + ("website" :components ("website-org" "website-static")))) + + +#+end_src +** Keybindings +Now we set up our keybindings for our applications: +#+begin_src emacs-lisp :tangle yes +(map! :leader + :desc "Open irc" + "i c" #'circe) +(map! :leader + :desc "Open audio manager" + "m m" #'emms) +(map! :leader + :desc "Open RSS feed reader" + "r s" #'elfeed) +(map! :leader + :desc "Open password manager" + "p w" #'ivy-pass) +#+end_src +** External Packages +we want to include some packages that don't come with doom emacs. +*** KBD-Mode +kbd-mode allows us to edit kmonad kbd files with syntax highlighting: +#+begin_src emacs-lisp :tangle yes +(use-package! kbd-mode) +#+end_src +*** Pinentry +We now set up pinentry for the pass program. We need to set the mode to loopback +in order to enable emacs to start itself as a pinentry program, and we need to allow +loopbacks in gpg-agent.conf. +#+begin_src emacs-lisp :tangle yes +(use-package! pinentry + :init (setq epa-pinentry-mode `loopback) + (pinentry-start)) +#+end_src diff --git a/config/elfeed.org b/config/elfeed.org new file mode 100644 index 0000000..bd71ffd --- /dev/null +++ b/config/elfeed.org @@ -0,0 +1,20 @@ +#+title: Elfeed Feed +#+author: Preston Pan +#+date: <2023-06-09 Fri> +#+description: My list of blogs, news sites, and channels. + +#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" /> +* Feed +** Blogs :blog: +Here are some interesting blogs, many of which are from substack. +*** https://astralcodexten.substack.com/feed +*** https://graymirror.substack.com/feed +*** https://lukesmith.xyz/index.xml :lukesmith: +** News :news: +*** https://rss.nytimes.com/services/xml/rss/nyt/Technology.xml +*** https://rss.nytimes.com/services/xml/rss/nyt/World.xml +*** https://rss.nytimes.com/services/xml/rss/nyt/Science.xml +*** https://rss.nytimes.com/services/xml/rss/nyt/AsiaPacific.xml +** Podcasts :podcast: +*** https://notrelated.xyz/rss :lukesmith: +** Misc. diff --git a/config/fish.org b/config/fish.org new file mode 100644 index 0000000..cb6e0af --- /dev/null +++ b/config/fish.org @@ -0,0 +1,55 @@ +#+title: Fish RC File +#+author: Preston Pan +#+date: <2023-06-09 Fri> +#+description: My fish shell configuration. + +#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" /> + +* Configuration +** Environment Variables +We define our path and also some environment variables. Since +I use guix as one of my package managers, I set GUIX_LOCPATH. +Also, I have a ~/.local/bin directory where I keep my scripts. +#+begin_src fish :tangle config.fish +fish_add_path -m ~/.local/bin +fish_add_path -m ~/gems/bin +fish_add_path -m ~/.local/share/gem/ruby/3.0.0/bin +set -x GUIX_LOCPATH $HOME/.guix-profile/lib/locale +set -x GEM_HOME "~/gems" +#+end_src +** Aliases +Now we define some aliases: +*** Git +We want to define aliases only if we call fish as an interactive shell. +We also define some common aliases for git commands: +#+begin_src fish :tangle config.fish +if status is-interactive + alias gcl="git clone" + alias gp="git push" + alias gpu="git pull" + alias gcm="git commit" +#+end_src +*** Pacman +Now we define aliases for commonly used pacman commands in the form +of the paru aur helper: +#+begin_src fish :tangle config.fish + alias p="paru -S" + alias pq="paru -sS" + alias pd="paru -Rncs" + alias syncweb="rsync -uvrP --delete-after website_html/ root@nullring.xyz:/var/www/ret2pop/" +#+end_src +*** Misc. +These are generally useful commands. Since I use mbsync, I wrote an alias +for it. +#+begin_src fish :tangle config.fish + alias c="clear" + alias l="ls -a" + alias mkdir="mkdir -pv" + alias syncmail="mbsync -c ~/.config/doom/mbsyncrc prestonpan" +#+end_src +** Vi Mode +Now we want to use vi mode because it is better. +#+begin_src fish :tangle config.fish + fish_vi_key_bindings +end +#+end_src diff --git a/config/index.org b/config/index.org new file mode 100644 index 0000000..1932468 --- /dev/null +++ b/config/index.org @@ -0,0 +1,27 @@ +#+title: Configurations +#+author: Preston Pan +#+date: <2023-06-10 Sat> +#+language: en +#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" /> +#+OPTIONS: broken-links:t +* What is this magic? +They're configurations for extremely obscure programs I use. Yeah, I know, I'm somewhat of a hipster myself. +** Configurations +Here is a list of my configurations for various programs: +#+begin_src shell :results output raw :exports both +for f in *; +do + if [[ "$f" == "index.org" ]]; then + continue + fi + printf -- "- [[file:$f][$f]]\n" +done +#+end_src + +#+RESULTS: +- [[file:doom.org][doom.org]] +- [[file:elfeed.org][elfeed.org]] +- [[file:fish.org][fish.org]] +- [[file:kmonad.org][kmonad.org]] +- [[file:qtile.org][qtile.org]] +- [[file:qutebrowser.org][qutebrowser.org]] diff --git a/config/kmonad.org b/config/kmonad.org new file mode 100644 index 0000000..4f38a86 --- /dev/null +++ b/config/kmonad.org @@ -0,0 +1,720 @@ +#+title: KMonad Configuration +#+author: Preston Pan +#+description: My Literate KMonad Configuration +#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" /> + +* Goals +** It should be the most efficient keyboard layout possible. +*** Ergonomic key positions and more frequent keys closer to home row. +*** Optimize the layers in such a way where we do not need to change layer frequently +however, layers are still needed because it is sometimes more economical to switch layers rather than to reach. + +* Configuration +** Initial Values +We must set our initial values for the configuration: +#+begin_src kbd :tangle config.kbd +(defcfg + input (device-file "/dev/input/by-path/platform-i8042-serio-0-event-kbd") + output (uinput-sink "My KMonad output" + "/usr/bin/sleep 1 && /usr/bin/setxkbmap -option compose:ralt") + cmp-seq ralt ;; Set the compose key to `RightAlt' + cmp-seq-delay 5 ;; 5ms delay between each compose-key sequence press + fallthrough true + allow-cmd true +) +#+end_src +you need to set your input device according to your hardware. +** defsrc +The defsrc block specifies the default layout that our keyboard uses. Note that when you are, +for example, using a 40% keyboard, some of these keys will not be present. +However, it is still syntactically valid and you will be able to use this +configuration. +#+begin_src kbd :tangle config.kbd +(defsrc + grv 1 2 3 4 5 6 7 8 9 0 - = bspc + tab q w e r t y u i o p [ ] \ + caps a s d f g h j k l ; ' ret + lsft z x c v b n m , . / rsft + lctl lmet lalt spc ralt rmet cmp rctl +) +#+end_src +Also note that the defsrc block is not used for custom key configuration. It +is used only to describe the shape of the current keyboard. +** defalias +defalias defines aliases for buttons. Buttons are used in deflayer blocks +in order to remap the keys. +#+begin_src kbd :tangle config.kbd +(defalias + num (layer-toggle numbers) + kil C-A-del +) +#+end_src +** deflayer +#+begin_src kbd :tangle config.kbd +#| -------------------------------------------------------------------------- + Necessary: at least 1 `deflayer` block + + It is also important to mention that the 'keymap' in KMonad is modelled as a + stack of layers (just like in QMK). When an event is registered we look in the + top-most layer for a handler. If we don't find one we try the next layer, and + then the next. + + Exactly what 'evaluates-to-a-button' will be expanded on in more detail below. + There are very many different specialist buttons in KMonad that we will touch + upon. However, for now, these 4 are a good place to begin: + + 1. Any keycode evaluates to a button that, on press, emits the press of that + keycode, and on release, emits the release of that keycode. Just a 'normal' + button. The exception is '\', which gets used as an escape character. Use + '\\' instead. Other characters that need to be escaped to match the literal + character are '(', ')', and '_'. + + 2. An @-prefixed name evaluates to an alias lookup. We named two buttons in + the `defalias` block above, we could now refer to these buttons using + `@num` and `@kil`. This is also why we only use alias-names no longer than + 3 characters in this tutorial. Also, note that we are already referencing + some aliases that have not yet been defined, this is not an issue. + + 3. The '_' character evaluates to transparent. I.e. no handler for that + key-event in this layer, causing this event to be handed down the layer + stack to perhaps be handled by the next layer. + + 4. The 'XX' character evaluates to blocked. I.e. no action bound to that + key-event in this layer, but do actually catch event, preventing any + underlying layer from handling it. + + Finally, it is important to note that the *first* `deflayer` statement in a + KMonad config will be the layer that is active when KMonad starts up. + + -------------------------------------------------------------------------- |# + + +(deflayer qwerty + grv 1 2 3 4 5 6 7 8 9 0 - = bspc + tab q w e r t y u i o p [ ] \ + esc a s d f g h j k l ; ' ret + lsft z x c v b n m , . / rsft + lctl lmet lalt spc @num rmet @sym rctl +) + +(deflayer numbers + _ _ _ _ _ _ _ _ _ _ _ _ _ _ + _ _ _ _ _ XX / 7 8 9 - _ _ _ + _ _ _ _ _ XX * 4 5 6 + _ _ + _ _ \( \) . XX 0 1 2 3 _ _ + _ _ _ _ _ _ _ _ +) + +#+end_src + +#+begin_src kbd :tangle config.kbd +#| -------------------------------------------------------------------------- + Optional: modded buttons + + Let's start by exploring the various special buttons that are supported by + KMonad by looking at 'modded' buttons, that is to say, buttons that activate + some kind of 'mod', then perform some button, and finally release that 'mod' + again. + + We have already seen an example of this style of button, our `kil` button is + one such button. Let's look at it in more detail: + C-A-del + + This looks like a simple declarative statement, but it's helpful to realize + that is simply syntactic sugar around 2 function calls. This statement is + equivalent to: + (around ctl (around alt del)) + + This highlights a core design principle in KMonad: we try to provide very + simple buttons, and then we provide rules and functions for combining them + into new buttons. Although note: still very much a work in progress. + + So, looking at this statement: + (around foo bar) + + Here, `around` is a function that takes two buttons and creates a new button. + This new button will, on a press, first press foo, then press bar, and on a + release first release bar, and then foo. Once created, this new button can be + passed to anything in KMonad that expects a button. + + We have already seen other examples of modded buttons, \(, \), *, and +. There + are no Keycodes for these buttons in KMonad, but they are buttons. They simply + evaluate to `(around lsft x)`. All shifted numbers have their corresponding + characters, the same is true for all capitals, and < > : ~ " | { } \_ + and ?. + + To wrap up 'modded-buttons', let's look back at C-A-del. We have 8 variants: + C- : (around lctl X) + A- : (around lalt X) + M- : (around lmet X) + S- : (around lsft X) + + Then RC-, RA-, RM-, and RS- behave exactly the same, except using the + right-modifier. + + These can be combined however you please: + C-A-M-S-x ;; Perfectly valid + C-% ;; Perfectly valid: same as C-S-5 + C-RC-RA-A-M-S-RS-m ;; Sure, but why would you? + + Also, note that although we provide special syntax for certain modifiers, + these buttons are in no way 'special' in KMonad. There is no concept of 'modifier'. + (around a (around b c)) ;; Perfectly valid + + -------------------------------------------------------------------------- |# + +(defalias + + ;; Something useful + cpy C-c + pst C-v + cut C-x + + ;; Something silly + md1 (around a (around b c)) ;; abc + md2 (around a (around lsft b)) ;; aB + md3 C-A-M-S-l + md4 (around % b) ;; BEWARE: %B, not %b, do you see why? +) + +#| -------------------------------------------------------------------------- + Optional: sticky keys + + KMonad also support so called "sticky keys". These are keys that will + behave as if they were pressed after just tapping them. This behaviour + wears off after the next button is pressed, which makes them ideal for + things like a quick control or shift. For example, tapping a sticky and + then pressing `abc' will result in `Abc'. + + You can create these keys with the `sticky-key' keyword: + + (defalias + slc (sticky-key 500 lctl)) + + The number after `sticky-key' is the timeout you want, in milliseconds. If + a key is tapped and that time has passed, it won't act like it's pressed + down when we receive the next keypress. + + It is also possible to combine sticky keys. For example, to + get a sticky shift+control you can do + + (defalias + ssc (around + (sticky-key 500 lsft) + (sticky-key 500 lctl))) + + -------------------------------------------------------------------------- |# + +;; Let's make both shift keys sticky +(defalias + sl (sticky-key 300 lsft) + sr (sticky-key 300 rsft)) + + +;; Now we define the 'tst' button as opening and closing a bunch of layers at +;; the same time. If you understand why this works, you're starting to grok +;; KMonad. +;; +;; Explanation: we define a bunch of testing-layers with buttons to illustrate +;; the various options in KMonad. Each of these layers makes sure to have its +;; buttons not overlap with the buttons from the other layers, and specifies all +;; its other buttons as transparent. When we use the nested `around` statement, +;; whenever we push the button linked to '@tst' (check `qwerty` layer, we bind +;; it to `rctl`), any button we press when holding `rctl` will be pressed in the +;; context of those 4 layers overlayed on the stack. When we release `rctl`, all +;; these layers will be popped again. +(defalias tst (around (layer-toggle macro-test) + (around (layer-toggle layer-test) + (around (layer-toggle around-next-test) + (around (layer-toggle command-test) + (layer-toggle modded-test)))))) + +(deflayer modded-test + _ _ _ _ _ _ _ _ _ _ _ _ _ _ + _ _ @md4 _ _ _ _ _ _ _ _ _ _ _ + _ _ @md1 @md2 @md3 _ _ _ _ _ _ _ _ + _ _ @cut @cpy @pst _ _ _ _ _ _ _ + _ _ _ _ _ _ _ _ +) + +#| -------------------------------------------------------------------------- + Optional: tap-macros + + Let's look at a button we haven't seen yet, tap-macros. + + `tap-macro` is a function that takes an arbitrary number of buttons and + returns a new button. When this new button is pressed it rapidly taps all its + stored buttons in quick succesion except for its last button, which it only + presses. This last button gets released when the `tap-macro` gets released. + + There are two ways to define a `tap-macro`, using the `tap-macro` function + directly, or through the #() syntactic sugar. Both evaluate to exactly the + same button. + + (tap-macro K M o n a d) + #(K M o n a d) + + If you are going to use a `tap-macro` to perform a sequence of actions inside + some program you probably want to include short pauses between inputs to give + the program time to register all the key-presses. Therefore we also provide + the 'pause' function, which simply pauses processing for a certain amount of + milliseconds. Pauses can be created like this: + + (pause 20) + P20 + + You can also pause between each key stroke by specifying the `:delay' keyword, + as well as a time in ms, at the end of a `tap-macro': + + (tap-macro K M o n a d :delay 5) + #(K M o n a d :delay 5) + + The above would be equivalent to e.g. + + (tap-macro K P5 M P5 o P5 n P5 a P5 d) + + The `tap-macro-release` is like `tap-macro`, except that it + waits to press the last button when the `tap-macro-release` + gets released. It might be useful when combined with a + footswitch that sends keyboard scan codes. + + (tap-macro-release i K M o n a d esc) + + WARNING: DO NOT STORE YOUR PASSWORDS IN PLAIN TEXT OR IN YOUR KEYBOARD + + I know it might be tempting to store your password as a macro, but there are 2 + huge risks: + 1. You accidentally leak your config and expose your password + 2. Anyone who knows about the button can get clear-text representation of your + password with any text editor, shell, or text-input field. + + Support for triggering shell commands directly from KMonad is described in the + command buttons section below. + + This concludes this public service announcement. + + -------------------------------------------------------------------------- |# + +(defalias + mc1 #(K M o n a d) + mc2 #(C-c P50 A-tab P50 C-v) ;; Careful, this might do something + mc3 #(P200 h P150 4 P100 > < P50 > < P20 0 r z 1 ! 1 ! !) + mc4 (tap-macro a (pause 50) @md2 (pause 50) c) + mc5 (tap-macro-release esc esc esc) + mc6 #(@mc3 spc @mc3 spc @mc3) +) + +(deflayer macro-test + _ @mc1 @mc2 @mc3 @mc4 @mc5 @mc6 _ _ _ _ _ _ _ + _ _ _ _ _ _ _ _ _ _ _ _ _ _ + _ _ _ _ _ _ _ _ _ _ _ _ _ + _ _ _ _ _ _ _ _ _ _ _ _ + _ _ _ _ _ _ _ _ +) + + +#| -------------------------------------------------------------------------- + Optional: layer manipulation + + You have already seen the basics of layer-manipulation. The `layer-toggle` + button. This button adds a layer to the top of KMonad's layer stack when + pressed, and removes it again when released. There are a number of other ways + to manipulate the layer stack, some safer than others. Let's go through all of + them from safest to least safe: + + `layer-toggle` works as described before, 2 things to note: + 1. If you are confused or worried about pressing a key, changing layers, and + then releasing a key and this causing issues: don't be. KMonad handles + presses and releases in very different ways. Presses get passed directly to + the stacked keymap as previously described. When a KMonad button has its + press-action triggered, it then registers a callback that will catch its + own release before we ever touch the keymap. This guarantees that the + button triggered by the press of X *will be* the button whose release is + triggered by the release of X (the release of X might trigger other things + as well, but that is besides the point.) + 2. If `layer-toggle` can only ever add and then necessarily remove 1 layer + from the stack, then it will never cause a permanent change, and is + perfectly safe. + + `layer-delay`, once pressed, temporarily switches to some layer for some + milliseconds. Just like `layer-toggle` this will never permanently mess-up the + layer stack. This button was initially implemented to provide some + 'leader-key' style behavior. Although I think in the future better solutions + will be available. For now this will temporarily add a layer to the top of the + stack: + (layer-delay 500 my-layer) + + `layer-next`, once pressed, primes KMonad to handle the next press from some + arbitrary layer. This aims to fill the same usecase as `layer-delay`: the + beginnings of 'leader-key' style behavior. I think this whole button will get + deleted soon, because the more general `around-next` now exists (see below) + and this is nothing more than: + (around-next (layer-toggle layer-name)) + Until then though, use `layer-next` like this: + (layer-next layer-name) + + `layer-switch`: change the base-layer of KMonad. As described at the top of + this document, the first `deflayer` statement is the layer that is active when + KMonad starts. Since `layer-toggle` can only ever add on and remove from the + top of that, it can never change the base-layer. The following button will + unregister the bottom-most layer of the keymap, and replace it with another + layer. + (layer-switch my-layer) + + This is where things start getting potentially dangerous (i.e. get KMonad into + an unusuable state until a restart has occured). It is perfectly possible to + switch into a layer that you can never get out of. Or worse, you could + theoretically have a layer full of only `XX`s and switch into that, rendering + your keyboard unuseable until you somehow manage to kill KMonad (without using + your keyboard). + + However, when handled well, `layer-switch` is very useful, letting you switch + between 'modes' for your keyboard. I have a tiny keyboard with a weird keymap, + but I switch into a simple 'qwerty' keymap shifted 1 button to the right for + gaming. Just make sure that any 'mode' you switch into has a button that + allows you to switch back out of the 'mode' (or content yourself restarting + KMonad somehow). + + `layer-add` and `layer-rem`. This is where you can very quickly cause yourself + a big headache. Originally I didn't expose these operations, but someone + wanted to use them, and I am not one to deny someone else a chainsaw. As the + names might give away: + (layer-add name) ;; Add a layer to the top of the stack + (layer-rem name) ;; Remove a layer by name (noop if no such layer) + + To use `layer-add` and `layer-rem` well, you should take a moment to think + about how to create a layout that will prevent you from getting into + situations where you enter a key-configuration you cannot get out of again. + These two operations together, however, are very useful for activating a + permanent overlay for a while. This technique is illustrated in the tap-hold + overlay a bit further down. + + + -------------------------------------------------------------------------- |# + +(defalias + + yah (layer-toggle asking-for-trouble) ;; Completely safe + nah (layer-add asking-for-trouble) ;; Completely unsafe + + ld1 (layer-delay 500 numbers) ;; One way to get a leader-key + ld2 (layer-next numbers) ;; Another way to get a leader key + + ;; NOTE, this is safe because both `qwerty` and `colemak` contain the `@tst` + ;; button which will get us to the `layer-test` layer, which itself contains + ;; both `@qwe` and `@col`. + qwe (layer-switch qwerty) ;; Set qwerty as the base layer + col (layer-switch colemak) ;; Set colemak as the base layer +) +(deflayer layer-test + @qwe _ _ _ _ _ _ _ _ _ _ @add _ @nah + @col _ _ _ _ _ _ _ _ _ _ _ _ @yah + _ _ _ _ _ _ _ _ _ _ _ _ _ + _ _ _ _ _ _ _ _ _ @ld1 @ld2 _ + _ _ _ _ _ _ _ _ +) + +;; Exactly like qwerty, but with the letters switched around +(deflayer colemak + grv 1 2 3 4 5 6 7 8 9 0 - = bspc + tab q w f p g j l u y ; [ ] \ + @xcp a r s |