Skip to content

Files

Latest commit

e73e5d0 · Feb 18, 2025

History

History
9457 lines (7860 loc) · 309 KB

system-config.org

File metadata and controls

9457 lines (7860 loc) · 309 KB

HOLD [#B] System: Gentoo Linux

  • State “HOLD” from “NEXT” [2021-01-31 Sun 23:08]
  • State “NEXT” from “HOLD” [2020-12-13 Sun 18:07]
  • State “HOLD” from “NEXT” [2020-12-11 Fri 16:00]
    Get full installation first

System upgrade:

su
eix-sync
emerge --update --ask --deep --keep-going --newuse --backtrack=50 world
haskell-updater
smart-live-rebuild
eselect kernel list
eselect kernel set 1
genkernel --menuconfig all
revdep-rebuild
eclean-pkg
eclean-dist
emerge --depclean --ask

Basic toolchain

System setup

The system is configured with full disk encryption, removed trusted Windows EFI key, and Linux kernel stored on USB drive. I used Sakaki’s guide: [[id:273a40e160b2bac3abab5110fbee51a2f9c3b0ac][[Wiki.Gentoo] User:Sakaki/Sakaki’s EFI Install Guide - Gentoo Wiki]]

[2023-01-07 Sat] Using more up-to-date https://wiki.gentoo.org/wiki/Full_Disk_Encryption_From_Scratch_Simplified#Create_partitions but also mixing it with some parts of Sakaki guide.

fstab

UUID=ea7aa879-3483-45c2-81cd-ecf40b49fb15		/				btrfs		defaults,noatime,discard			0 1
UUID=adfc4270-9b46-4fc3-b097-6882e910b37f		none				swap		defaults,noatime,discard			0 0
UUID=d8bb8479-30b3-4b6b-acaa-43a735ddd020		/home				btrfs		defaults,noatime,discard			0 2
efivarfs						/sys/firmware/efi/efivars	efivarfs	rw,nosuid,nodev,noexec,relatime			0 0
UUID=3244-9D07						/boot				vfat		noauto,noatime					1 2

Portage

Binary package repository

See https://wiki.gentoo.org/wiki/Gentoo_Binary_Host_Quickstart

[binhost]
priority = 9999
sync-uri = https://mirror.bytemark.co.uk/gentoo/releases/amd64/binpackages/23.0/x86-64/

Main package repository

[DEFAULT]
main-repo = gentoo

[gentoo]
location = /var/db/repos/gentoo
sync-type = webrsync
#sync-type = rsync
sync-uri = rsync://rsync.gentoo.org/gentoo-portage
auto-sync = yes

sync-rsync-verify-jobs = 1
sync-rsync-verify-metamanifest = yes
sync-rsync-verify-max-age = 24
sync-openpgp-key-path = /usr/share/openpgp-keys/gentoo-release.asc
sync-openpgp-keyserver = hkps://keys.gentoo.org
sync-openpgp-key-refresh-retry-count = 40
sync-openpgp-key-refresh-retry-overall-timeout = 1200
sync-openpgp-key-refresh-retry-delay-exp-base = 2
sync-openpgp-key-refresh-retry-delay-max = 60
sync-openpgp-key-refresh-retry-delay-mult = 4
sync-webrsync-verify-signature = yes

Eclass docs: eclass-manpages

make.conf

Compiler flags
(string-to-number (string-remove-suffix "\n" (shell-command-to-string "nproc")))

Do not use literally all the CPUs for compilation to keep the laptop usable even during upgrades.

(- <<numcpus()>> 2)
# C, C++ and FORTRAN options for GCC.
MAKEOPTS="-j<<numcpusreduced()>> -l<<numcpusreduced()>>"
EMERGE_DEFAULT_OPTS="--jobs=<<numcpusreduced()>> --load-average=<<numcpusreduced()>>"
COMMON_FLAGS="-march=native -O3 -pipe"
CFLAGS="${COMMON_FLAGS}"
CXXFLAGS="${COMMON_FLAGS}"
FCFLAGS="${COMMON_FLAGS}"
FFLAGS="${COMMON_FLAGS}"
Branch

Use the ‘unstable’ branch

ACCEPT_KEYWORDS="~amd64"
Global use flags
  • vulkan: graphic hardware acceleration
USE="$USE X truetype bash-completion unicode emacs fontconfig gnuplot latex libnotify syslog udev pulseaudio"
USE="$USE -qt4 -qt3support -dvd -dvdr -firefox -gnome"
USE="$USE jit"
USE="$USE elogind -consolekit -systemd"
USE="$USE cjk"
USE="$USE vulkan"
CPU_FLAGS_X86="aes avx avx2 f16c fma3 mmx mmxext pclmul popcnt sse sse2 sse3 sse4_1 sse4_2 ssse3"
Use binary packages unless custom use flags are selected
EMERGE_DEFAULT_OPTS="$EMERGE_DEFAULT_OPTS --getbinpkg"
Portage directories
PORTDIR="/var/db/repos/gentoo"
DISTDIR="/var/cache/distfiles"
PKGDIR="/var/cache/binpkgs"
Language for build output
# This sets the language of build output to English.
# Please keep this setting intact when reporting bugs.
LC_MESSAGES=C
Emerge logging
# Turn on logging - see http://gentoo-en.vfose.ru/wiki/Gentoo_maintenance
PORTAGE_ELOG_CLASSES="info warn error log qa"
# Echo messages after emerge, also save to /var/log/portage/elog
PORTAGE_ELOG_SYSTEM="echo save"

# Ensure elogs saved in category subdirectories.
# Build binary packages as a byproduct of each emerge, a useful backup
# Enable binary packages
FEATURES="split-elog buildpkg binpkg-request-signature"
Video cards
# Settings for X11
VIDEO_CARDS="intel i965"
INPUT_DEVICES="libinput"
Gentoo mirrors
GENTOO_MIRRORS="https://mirror.yandex.ru/gentoo-distfiles/ http://gentoo.aditsu.net:8000/ http://mirror.rise.ph/gentoo http://ftp.daum.net/gentoo/ http://ftp.kaist.ac.kr/pub/gentoo/ https://ftp.lanet.kr/pub/gentoo/"
Languages
LINGUAS="en ru uk zh-CN"
L10N="en-US ru uk zh-CN ar zh de es fr en"

eclean-kernel : Clean old kernel dirs

app-admin/eclean-kernel

smart-live-rebuild : Rebuild 9999 packages

app-portage/smart-live-rebuild

Linux kernel

sys-kernel/gentoo-sources

Firmware for Linux

sys-kernel/linux-firmware linux-fw-redistributable no-source-code

Disk encryption support

sys-fs/cryptsetup

Building and installing the kernel (genkernel and grub)

#boot #EFI #grub #genkernel #not_loading

[2024-01-15 Mon] EFI requires VRAM (BIOS memory) and apparently genkernel keeps adding things there. Now, I ran out of VRAM causing grub-install to fail silently (genkernel did not report anything, slurping “No space left” error reported by grub). I had to reset EFI keys from BIOS and later clear VRAM from BIOS prompt. Now fixed.

Installing after update

genkernel --menuconfig all
grub-mkconfig -o /boot/grub/grub.cfg

[2024-03-23 Sat] Need to install installkernel (it generates vmlinuz-x.y.z in /boot) manually according to Gentoo news.

sys-kernel/installkernel
sys-kernel/genkernel
sys-boot/grub
# kernel signature and EFI trust certificate manipulation
app-crypt/efitools
LUKS="yes"
LVM="yes"
DOLVM="yes"
ZFS="no"
GPG="yes"
BOOTLOADER="grub2"
NOCOLOR="false"
GK_SHARE="${GK_SHARE:-/usr/share/genkernel}"
CACHE_DIR="/var/cache/genkernel"
DISTDIR="${GK_SHARE}/distfiles"
LOGFILE="/var/log/genkernel.log"
LOGLEVEL=1
DEFAULT_KERNEL_SOURCE="/usr/src/linux"
SAVE_CONFIG="yes"
USE="$USE lvm"
rc-update add lvm default
sys-boot/grub:2 device-mapper

[2024-03-12 Tue] Observing SSD errors in dmesg log, like in https://unix.stackexchange.com/questions/470778/nvme-missing-or-invalid-subnqn-field Disabling SSD power saving as suggested in the answer.

[2024-10-11 Fri] RAM failures corrupted hard drive. Disabling (memmap) faulty memory address range according to memtest.

GRUB_DISTRIBUTOR="Gentoo"
GRUB_CMDLINE_LINUX="memmap=300M\\\$4200M dolvm crypt crypt_root=UUID=b074fbfd-65b3-4830-971c-0dd07b8977b5 root_trim=yes root_key=luks-key.gpg nvme_core.default_ps_max_latency_us=0"
GRUB_TIMEOUT="1"

Pip

dev-python/pip

System logging sysklogd

app-admin/sysklogd
rc-update add sysklogd default

cron

sys-process/cronie
rc-update add cronie default
SHELL=/bin/bash
PATH=/bin:/usr/bin:/home/yantar92/.local/bin
HOME=/home/yantar92
DISPLAY=:0

Hibernation \ Suspend (elogind)

sys-auth/elogind

[2021-01-23 Sat] pm-utils is removed from Gentoo because upstream is abandoned. elogind should be sufficient though.

Note that it does not work automatically. Need to change ACPI(d) config.

Lock screen after resuming from suspend

case $1 in
    resume)
	export DISPLAY=:0
        sudo -u yantar92 xtrlock
esac

ntfs mount using ntfs3g

sys-fs/ntfs3g

USB automount with udiskie

[[id:github_coldfix_coldf_udisk_autom_remov_media][coldfix [Github] Coldfix Udiskie: Automounter for Removable Media]]
sys-fs/udiskie
udiskie &

ACPI(d)

sys-power/acpid

Must run acpid service

rc-update add acpid default

Default handler

# $Header: /etc/acpi/default.sh                          Exp $
# $Author: (c) 2012-2014 -tclover <tokiclover@dotfiles.> Exp $
# $License: MIT (or 2-clause/new/simplified BSD)         Exp $
# $Version: 2014/12/24 21:09:26                          Exp $
#
 
log() { logger -p daemon "ACPI: $*"; }
uhd() { log "event unhandled: $*"; }
 
set $*
group=${1%/*}
action=${1#*/}
device=$2
id=$3
value=$4
 
[ -d /dev/snd ] && alsa=true || alsa=false
[ -d /dev/oss ] && oss=true || oss=false
amixer="amixer -q set Master"
ossmix="ossmix -- vmix0-outvol"
 
case $group in
    # ac_adapter)
    # 	case $value in
    # 		*0) log "switching to power.bat power profile"
    # 			hprofile power.bat;;
    # 		*1) log "switching to power.adp power profile"
    # 			hprofile power.adp;;
    # 		*) uhd $*;;
    # 	esac
    # 	;;
    # battery)
    # 	case $value in
    # 		*0) log "switching to power.adp power profile"
    # 			hprofile power.adp;;
    # 		*1) log "switching to power.adp power profile"
    # 			hprofile power.adp;;
    # 		*) uhd $*;;
    # 	esac
    # 	;;
    button)
	case $action in
	    lid)
		case "$id" in
		    close) loginctl suspend;;
		    open) :;;
		    *) uhd $*;;
		esac
		;;
	    power) shutdown -H now;;
	    sleep) loginctl hibernate;;
	    #			mute) 
	    #				$alsa && $amixer toggle;;
	    #			volumeup) 
	    #				$alsa && $amixer 3dB+
	    #				$oss && $ossmix +3;;
	    #			volumedown) 
	    #				$alsa && $amixer 3dB-
	    #				$oss && $ossmix -3;;
	    *) uhd $*;;
	esac
	;;
    # cd)
    # 	case $action in
    # 	    play) :;;
    # 	    stop) :;;
    # 	    prev) :;;
    # 	    next) :;;
    # 	    *) uhd $*;;
    # 	esac
    # 	;;
    # jack)
    # 	case $id in
    # 	    *plug) :;;
    # 	    *) uhd $*;;
    # 	esac
    # 	;;
    video)
	case $action in
	    displayoff) :;;
	    brightnessup) log "Increasing brightness..."
			  /etc/acpi/actions/Fn-brightnessup.sh;;
	    brightnessdown) log "Decreasing brightness..."
			    /etc/acpi/actions/Fn-brightnessdown.sh;;
	    *) uhd $*;;
	esac
	;;
    *) uhd $*;;
esac
 
unset alsa oss amixer ossmix group action device id

Default screen brightness (on startup)

SUBSYSTEM=="backlight", ACTION=="add", KERNEL=="intel_backlight", ATTR{brightness}="700"

Screen brightness key bindings

# Set the static decrement value.  Keep in mind that this will 
# be done twice. 
DecVal=150
LowDecVal=4 
 
# Set the Minimum we will accept. 
MinVal=0 
 
# Get the current brightness value. 
#CurrVal=$(cat /sys/class/backlight/intel_backlight/brightness); 
read -r CurrVal < "/sys/class/backlight/intel_backlight/brightness"
 
# Set the new value minus the decrement value. 
NewVal=$(($CurrVal<=$DecVal?($CurrVal - $LowDecVal):($CurrVal - $DecVal))); 
echo $NewVal 
 
# Set it to the threshold of the min value. 
ThresholdVal=$(($NewVal>$MinVal?$NewVal:$MinVal)) 
echo $ThresholdVal 
 
# Set the new value directly. 
echo -n $ThresholdVal > /sys/class/backlight/intel_backlight/brightness 
 
logger "[ACPI] brightnessdown |$CurrVal<nowiki>| |</nowiki>$NewVal| |$ThresholdVal|"
# Set the static increment value.  Keep in mind that this will 
# be done twice. 
IncVal=150
LowIncVal=4 
 
# Get the Maximum value for use. 
#MaxVal=$(cat /sys/class/backlight/intel_backlight/max_brightness); 
read -r MaxVal < "/sys/class/backlight/intel_backlight/max_brightness"
 
# Get the current brightness value. 
#CurrVal=$(cat /sys/class/backlight/intel_backlight/brightness); 
read -r CurrVal < "/sys/class/backlight/intel_backlight/brightness"
 
# Set the new value minus the decrement value. 
NewVal=$(($CurrVal<$IncVal?($CurrVal + $LowIncVal):($CurrVal + $IncVal))); 
echo $NewVal 
 
# Set it to the threshold of the max value. 
ThresholdVal=$(($NewVal<$MaxVal?$NewVal:$MaxVal)) 
echo $ThresholdVal 
 
# Set the new value directly. 
echo -n $ThresholdVal > /sys/class/backlight/intel_backlight/brightness 
 
logger "[ACPI] brightnessup |$CurrVal| |$NewVal| |$ThresholdVal|"

gpg

I had to copy pubring.gpg, pubring.kbx, trustdb.gpg, secring.gpg, and private-keys-v1.d

store them in attachment (personal)END

Yubikey settings

Solving the issue with no card detection by gnupg

Need to enable libusb explicitly (otherwise cannot detect the Yubikey):

app-crypt/gnupg usb
Configuration tool: yubikey-personalization-gui
sys-auth/yubikey-personalization-gui
Graphical password dialogue (pinentry)

[[id:67113260e2a4d7c02a3868d0198dd7651949524b][[Wiki.Gentoo] GnuPG - Gentoo Wiki]]: Need pinentry

app-crypt/pinentry

Use GTK dialogue and allow password keyring

app-crypt/pinentry gtk gnome-keyring
eselect pinentry set pinentry-gnome3

Password cache

# Set the default cache time to 1 day.
default-cache-ttl       86400
default-cache-ttl-ssh   86400

# Set the max cache time to 30 days.
max-cache-ttl           2592000
max-cache-ttl-ssh       2592000

ssh

I need to copy over my ssh key id_rsa and id_rsa.pub

store them in safe placeEND

ssh configuration

-----BEGIN PGP MESSAGE-----

hQEMA3SKBiQ2zhL6AQf/Uqd6izxCdSYkczdRaHe/yZN4bePtUtG1OQppH+rPUXjr nevVrjNzH9rYF50odXeaWlxcuzInOjjCYOcTgRmminiu21VAmCnO5f5zbnd2E1ZK SaYVUoQ/hhpSDPSxRO+Bo2OgHAEJ+kmqY+wPyRIW/phk/pNugg72fVm8sG6pMQ6B j9RTTBsSPXH2t+fA0wLEM2DyGSBjG3+bTP9JQHhTtKQuQmQIXocX3C0Ugd4aXX2N BXqAvDiAnxpbvsSzynfaoJvb1F5b1HKUb7eqRd4aYBM0iRZ1I1a57hJZTv0ondDE jppW81UjQCkzFXwgxNzKkpjYU15MinlbGdMW1zfJx9LAawHBigL9tYs0fIOXVZAU imhWCr9c4KIRIW0C3Dezy0UxSj+CAkRDbqLwD9TkgZingGwmjKk6m4H3aznyAn4A X66G9PppQgqGpXIUqTrROiF7KNghXan3T6Lm6qdTMbEYN4hBGjTgt+DScYgE2Yhu 3T2UC6iiQhbPtY8OTNvv9sb1qknYpPOXomgcd9onZSjLXafa73x8lYhpwIeVscLk oeX6wkYL1Y0tackjyqzQi2jhPpNWdvzp14jasPzv2K3wzi238zRm0FArD1wBg/Dt nnypD0DJs9t4RpbUSL5D91bKiE+alNVLHM2LTFqjkKTPD8j73Vzpf3wHFkKQwgSl DaU/KqmVouyC1FcAzM/HWCmdOaN+4EsU/3yqR3KfJa47SJFXWdWR0n5rDOmg =TrIU -----END PGP MESSAGE-----

sshfs: Mount remove ssh as a folder

net-fs/sshfs

lsof

Useful to check what is using usb drive

sys-process/lsof

alsa-utils

media-sound/alsa-utils

Card reader

[[id:273a40e160b2bac3abab5110fbee51a2f9c3b0ac][[Wiki.Gentoo] User:Sakaki/Sakaki’s EFI Install Guide - Gentoo Wiki]]: There is issue with some card readers when kernel modules are available, but the hardware is not loaded:

Although the necessary kernel options (MMC_SDHCI and MMC_SDHCI_PCI) for this card are modularized in the minimal install kernel, there is a bug impacting the CF-AX3 (and many other machines) which prevents correct initialization when a card is inserted. To fix this, still in the second terminal, issue:

options sdhci debug_quirks=0x40

Network

WiFi configuration

-----BEGIN PGP MESSAGE-----

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 mpsKI0LtjQ== =5cgj -----END PGP MESSAGE-----

User access to wpa_supplicant gui + TKIP support for common WiFi routers in China

I need to grant superuser access for calling wpa_gui if I want to be able to change networks on-the-fly. [2022-04-03 Sun] After update, I am suddenly unable to connect to all the password-protected WiFi in the area (except my phone). The WiFi is working, but there is no connection. Seems to be crypto support issue with changed defaults after update.

net-wireless/wpa_supplicant qt5 tkip

wireless-tools

net-wireless/wireless-tools

VPN

Kernel configuration

[[id:bdff11995982e871875b024ba60a988d82aa66c8][[Wiki.Gentoo] OpenVPN - Gentoo Wiki]] [[id:f7a56a1f03ef6bebbec21cea94a22a6a7d39ee11][[Wiki.Gentoo] Wireguard - Gentoo Wiki]]

VPN service

-----BEGIN PGP MESSAGE-----

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 =K701 -----END PGP MESSAGE-----

Docker

Installing following [[id:6ac6c92eec714e52a7fb5cf0e1a70ef62b789b83][[Wiki.Gentoo] Docker - Gentoo Wiki]]

app-containers/docker
app-containers/docker-cli
rc-update add docker default
app-containers/docker btrfs cli

Podman - alternative to docker

app-containers/podman

Also, need to manually install config and policy files.

cp /etc/containers/policy.json.example /etc/containers/policy.json
cp /etc/containers/registries.conf.example /etc/containers/registries.conf
# see https://forums.gentoo.org/viewtopic-t-1145234-start-0.html
sed -i -e 's|location="quay.io/libpod"|location="docker.io/library"|' /etc/containers/registries.conf
dev-ruby/rake

7z (p7zip)

app-arch/p7zip

Calculator (bc)

sys-devel/bc

sudo

-----BEGIN PGP MESSAGE-----

hQEMA3SKBiQ2zhL6AQf/UmaANBYo8ctQHfyL1h0ZGAUyb+zkbg1m2H4/TiMmLSxm 3n2VxTiUSdvszWuSVCg388icDfp5HOB4Bd4fL8TJ6xFAh+LVejvZvXEVwp1SKW2E U7qyVpoyld/UA7hi/ueO2eS6eWve15Cc2m5hw4oGHPTnrxZ3rRl4T7PSU7jcihCd ZeUj5MbANuvx+mQ0VlNLEY0C9FHNPlp+d6nH5XcYCHmmTdz/KsKNtzCMgOBGvSrH 7aHwfa5vBXIhGIGlZ7uVhDf0h2NLHf5kIB1SV2T1ea2XrUv2MlRE/XxleHs/sxPf SO6Oclt32ftDDzjVoUWt44dW+/O0YFUp2wWN03CvMtLA0QGmYDh68gb6/otFZCv0 kAiD4UVuIrulfTWCr5akvuN7dpMOaV9iM7v2DmV5UFSsJzDBTE+xsejlEtLFekDO +9bfvBWkpvl2uEPq2ukgaCWCWwdnLP50PIs/ZCKzo2oXxaH4DdUwCv/a6W93UjXw iNQm8iIwVabR6XJVQaj97/k0k7Eja56zs/tb/2CBa22miIpr8GOxc4rUOjhJoGAr 6m4mb8OfdOysB2FDXmeM//YK/pXBfrmxl3JQxSD2J3yaYkBpeb5yH5vm1omelGWy iEy5YHiVJ2CAN/hiN9YwI4jazPR04k+0McYzDt6P92Q9PxsHYl7NTKnpGtwakSZW 9lfJtflKM9oE+eThyDmuIehER/QiywYlhJAEET5hWg1rYf4RAwkGb+b3ug4Fl0Di hskd2bDSEcPUE30mzsEnL70nEZgU+eayP9qYRYR9DsJOajwKACsFAemMRw9Z3iBK meGrojRTkmhMViEBO0t3Mj4eTX1U7fhcY79iJi5BI39kbO+1qrSHEk/WgjTv4EMo BFav =Bv4S -----END PGP MESSAGE-----

btrfs-progs

sys-fs/btrfs-progs

restic backup

app-backup/restic

GUI

X window system

Wmctrl

x11-misc/wmctrl

Xdotool

x11-misc/xdotool

Xprop

x11-apps/xprop

Xtrlock

x11-misc/xtrlock

setxkbmap

x11-apps/setxkbmap

notification-daemon

x11-libs/libnotify

Liberation fonts

These are needed for high-resolution fonts in most apps.
media-fonts/liberation-fonts

X window defaults for all the apps

Defaults same as in bash

source .profile

Xorg

x11-base/xorg-server

Display manager: LightDM

Setup is trivial https://wiki.gentoo.org/wiki/LightDM

x11-misc/lightdm
gui-libs/display-manager-init
rc-update add dbus default
rc-update add display-manager default
# We always try and start the DM on a static VT. The various DMs normally
# default to using VT7. If you wish to use the display-manager init
# script, then you should ensure that the VT checked is the same VT your
# DM wants to use.
# We do this check to ensure that you haven't accidentally configured
# something to run on the VT in your /etc/inittab file so that
# you don't get a dead keyboard.
CHECKVT=7

# What display manager do you use ?
#     [ xdm | greetd | gdm | sddm | gpe | lightdm | entrance ]
# NOTE: If this is set in /etc/rc.conf, that setting will override this one.
DISPLAYMANAGER="lightdm"

Window manager: Awesome WM

Installation

https://wiki.gentoo.org/wiki/Awesome

x11-wm/awesome

Also need to use X flag for dbus. Otherwise, cannot connect to awesome-client.

x11-wm/awesome dbus doc
sys-apps/dbus X

Configuration

this should be reviewed and compared with modern API and example configEND
Includes

Vicious module: https://github.com/vicious-widgets/vicious.git

write ebuild and install properly (and delete local folder)ENDit also requires iwconfig command (wireless-tools) for wifi widgetEND
-- Standard awesome library
gears = require("gears")
awful = require("awful")
require("awful.autofocus")
-- Widget and layout library
wibox = require("wibox")
-- Theme handling library
beautiful = require("beautiful")
-- Notification library
naughty = require("naughty")
menubar = require("menubar")
vicious = require("vicious")
Error handling
-- Check if awesome encountered an error during startup and fell back to
-- another config (This code will only ever execute for the fallback config)
if awesome.startup_errors then
   naughty.notify({ preset = naughty.config.presets.critical,
		    title = "Oops, there were errors during startup!",
		    text = awesome.startup_errors })
end

-- Handle runtime errors after startup
do
   local in_error = false
   awesome.connect_signal("debug::error", function (err)
			     -- Make sure we don't go into an endless error loop
			     if in_error then return end
			     in_error = true

			     naughty.notify({ preset = naughty.config.presets.critical,
					      title = "Oops, an error happened!",
					      text = tostring(err) })
			     in_error = false
					  end)
end
Theme
create git repo for the theme and unify itENDcopy wallpaper to the theme itselfEND
-- Themes define colours, icons, font and wallpapers.
home_dir = "/home/yantar92/"
beautiful.init(home_dir .. ".config/awesome/themes/powerarrowf/theme.lua")

-- {{ These are the power arrow dividers/separators }} --
arr1 = wibox.widget.imagebox()
arr1:set_image(beautiful.arr1)
Widgets
Time and date
--{{-- Time and Date Widget }} --
tdwidget = wibox.widget.textbox()
--local strf = '<span font="' .. beautiful.font .. '" color="#EEEEEE" background="#777E76">%a %b %d %H:%M</span>'
local strf = '<span font="' .. beautiful.font .. '"> %a %b %d %H:%M </span>'
vicious.register(tdwidget, vicious.widgets.date, strf, 20)
clockicon = wibox.widget.imagebox()
clockicon:set_image(beautiful.clock)
Battery

The widget show battery charge and status (charging / discharging). Also, it notifies the user if the battery level drops too low and automatically hibernates the system in such a case. The hibernation requires superuser access and is allowed in sudo configuration:

STATUS="$(cat /sys/class/power_supply/BAT0/uevent | grep POWER_SUPPLY_POWER_NOW | cut -d= -f2)"
STATUS2="$(cat /sys/class/power_supply/BAT0/uevent | grep POWER_SUPPLY_POWER_STATUS | cut -d= -f2)"
if [[ "$STATUS2" == "Charging" ]]; then
    echo 0;
else
    [[ "$STATUS" == "0" ]] && echo 0 || echo 1
fi
baticon = wibox.widget.imagebox()
vicious.register(baticon, vicious.widgets.bat, function(widget, args)
		      stdout=assert(io.popen("battery_status.sh", 'r'))
		      local file_adapter = stdout:read('*all')
		      stdout:close()
		      if tonumber(file_adapter)==1 then
			 baticon:set_image(beautiful.baticon)
		      else
			 baticon:set_image(beautiful.chargeicon)
		      end
						 end, 5, "BAT0")

batwidget = wibox.widget.textbox()
vicious.register(batwidget, vicious.widgets.bat, function(widget, args)
		      textcolor=beautiful.fg_normal
		      stdout=assert(io.popen("battery_status.sh", 'r'))
		      local file_adapter = stdout:read('*all')
		      stdout:close()
		      if args[2] < 10 then
			 textcolor="#FF0000"
		      end
		      if args[2]<5 and tonumber(file_adapter)==1 then
			 naughty.notify({ preset = naughty.config.presets.critical,
					  title = "Low battery level! "..args[2]..'%',
					  text = '--------------------------------------------------' })
		      end
		      if args[2]<4 and tonumber(file_adapter)==1 then
			 awful.util.spawn("sudo loginctl hibernate")
		      end
		      return '<span font="'.. beautiful.font ..'"><span font="'.. beautiful.font ..'" color="'..textcolor..'">'..args[2]..'%</span></span>  ' end, 30, "BAT0" )
Net ssid widget

We need to find the WiFi interface name:

iwconfig 2>&1 | grep -oE "wlp[^ ]+"
netwidget = wibox.widget.textbox()
vicious.register(netwidget, vicious.widgets.wifi, function(widget, args)
		    cur_ssid=args["{ssid}"];
		    if cur_ssid:len() > 15 then
		       cur_ssid=cur_ssid:sub(1,12).."..."
		    end
		    if cur_ssid=="N/A" then
		       cur_ssid='<span font="'.. beautiful.font ..'" color="#FF0000">N/A</span>'
		    end
		    return '<span font="'.. beautiful.font ..'"><span font ="'.. beautiful.font ..'">'..cur_ssid..' </span></span>' end, 10, "<<wifi-name()>>")
Wifi widget
neticon = wibox.widget.imagebox()
vicious.register(neticon, vicious.widgets.wifi, function(widget, args)
		    local sigstrength = tonumber(args["{linp}"])
		    if sigstrength > 69 then
		       neticon:set_image(beautiful.nethigh)
		    elseif sigstrength > 40 and sigstrength < 70 then
		       neticon:set_image(beautiful.netmedium)
		    elseif sigstrength > 0 then
		       neticon:set_image(beautiful.netlow)
		    else
		       neticon:set_image(beautiful.netno)
		    end
						end, 10, '<<wifi-name()>>')
Volume

This requires amixer (alsa-utils)

should be dependency for viciousEND
volume = wibox.widget.textbox()
vicious.register(volume, vicious.widgets.volume,
		 '<span font="'.. beautiful.font ..'"><span font="'.. beautiful.font ..'">$1% </span></span>', 1, "Master")
volumeicon = wibox.widget.imagebox()
vicious.register(volumeicon, vicious.widgets.volume, function(widget, args)
		    local paraone = tonumber(args[1])
		    if args[2] == "🔈" or paraone == 0 then
		       volumeicon:set_image(beautiful.mute)
		    elseif paraone >= 67 and paraone <= 100 then
		       volumeicon:set_image(beautiful.volhi)
		    elseif paraone >= 33 and paraone <= 66 then
		       volumeicon:set_image(beautiful.volmed)
		    else
		       volumeicon:set_image(beautiful.vollow)
		    end
						     end, 1, "Master")
CPU
cpuwidget = wibox.widget.textbox()
vicious.register(cpuwidget, vicious.widgets.cpu,
		 function(widget, args)
		    text="";
		    if args[1] < 10 then
		       text=args[1].."%   "
		    elseif args[1] < 100 then
		       text=args[1].."% "
		    else
		       text=args[1].."%"
		    end
		    return '<span font="'.. beautiful.font ..'"><span font="'.. beautiful.font ..'">'..text..' </span></span>'
		 end, 1)
cpuicon = wibox.widget.imagebox()
cpuicon:set_image(beautiful.cpuicon)
Memory
memwidget = wibox.widget.textbox()
vicious.register(memwidget, vicious.widgets.mem, '<span font="'.. beautiful.font ..'"><span font="'.. beautiful.font ..'">$1% </span></span>', 1)
memicon = wibox.widget.imagebox()
memicon:set_image(beautiful.mem)
VPN status

-----BEGIN PGP MESSAGE-----

hQEMA3SKBiQ2zhL6AQf/S07KGLg9U5UI6Y8p7zLjjNdqxxmMVIXAwb8jY7KuPsCV PZQtFiW9CKs5zQW7dbqptog5E6YOnfkd3+QRVWihlZgnWUpLf6N/rdpMZdY2Q6Qv V3Vo5dml1mkd543pcKg0GiVJSmUiU7qsEjLKEHcC4oWt2io4G/e9krSLCW8kVISZ 6Y/0hUHvMhuIoCJ1k/AHX81ufaqahQ0FJ0gE1GsXIAVG3xDkN5aEUUvID4M4wJKQ /k1+DKSO8zwAX+fWiJh00rV/cxPnmne85FgWrKK1v0Dt3KKj57OQP7HNp7av8PZU slVaG2wv20UEaY23dah8XYgzjLPeBEx52oRxp3kKsdLBLAGUDaHW2B1+9YUT6R+n hyoAduvFvvvGCcWHBvsN03u+pLLt8D/Lz6il7PVFyskvXR93knRzcBMJnP1nzq0c dAjan4Gq+PwhV0U4jhJaCXln+9a7MXiQQQ37Eu9dSd7HBE8M/+1W29EeIVYbCl94 V7lLu/T4wFwzOGBa7DvdRweuXCvuath4UfwH1rkHJbtkqfagfq1tUuXZ9jHZbCto TyKGJDjZkUJyuq+TDLiVzXCUvrrK4Q/HbHyCXQT/b0HIJ2yFODO1+w4qjU1XVfKn Vc9K9ax9/n7wErrs4unzdDozJuPSP+90l+Wx+OPlys2+XVGFOm9V9TztlVsXAmmN 3oACicnRUMYaUXebhcNt8NU7MB0v2iKCOK3qKvvBYkTtwGtUi/FLtEwe6tG7fV/2 4ZbdaxL+1DNschUzAbHIXl2xN+NxBAOvrFqfy8RgErMuIANu3cuz5OlH6w4qndlm GKnqZ001zmo9lcl1MStpZTYXJ0XgjV+4Ll8RRpI8UhLJS7HE93KZe0zexfvONNQh Gap0LopCi6H/98HBucy3+uvcviRnP2qUhrHxLGTutMRg5KfjTviM9yRjXLbc3X7J 72uF7ruDiUucV4mtEKZi3394L1EvFlMDblmGByotz+wr/TN+1+L7bd2accc44A== =doJG -----END PGP MESSAGE-----

Externally controlled widgets
--{{--|Org status message |----------------
orgstatus = wibox.widget.textbox()
orgstatus:set_text("...")

--{{--|Current balance of important money/time |----------------
balance = wibox.widget.textbox()
balance:set_text("N/A")

--spacing
spacing = wibox.widget.textbox()
spacing:set_text(" ")
System tray

I use vertical system tray.

-- systray
mysystray = wibox.widget.systray()
mysystray:set_horizontal(false)
Diagonal icon spacer
diagicon = wibox.widget.imagebox()
diagicon:set_image(beautiful.diag)
Default apps
terminal = "kitty"
editor = os.getenv("EDITOR") or "nano"
editor_cmd = "emacsclient -c -n -e '(switch-to-buffer nil)'"
Modkey
-- Default modkey.
-- Usually, Mod4 is the key with a logo between Control and Alt.
-- If you do not like this or do not have such a key,
-- I suggest you to remap Mod4 to another key using xmodmap or other tools.
-- However, you can use another modifier like Mod1, but it may interact with others.
modkey = "Mod4"
Tiling layouts
-- Table of layouts to cover with awful.layout.inc, order matters.
awful.layout.layouts = {
   awful.layout.suit.tile.bottom,
   awful.layout.suit.tile,
   awful.layout.suit.tile.left,
   awful.layout.suit.tile.top,
   --    awful.layout.suit.spiral,
   awful.layout.suit.spiral.dwindle,
   awful.layout.suit.fair,
   awful.layout.suit.fair.horizontal,
   --    awful.layout.suit.max,
   --    awful.layout.suit.max.fullscreen,
   -- awful.layout.suit.magnifier,
   -- awful.layout.suit.corner.nw,
   -- awful.layout.suit.corner.ne,
   -- awful.layout.suit.corner.sw,
   -- awful.layout.suit.corner.se,
   awful.layout.suit.floating,
}
Menubar config
menubar.utils.terminal = terminal -- Set the terminal for applications that require it
Conky integration
function get_conky()
   local clients = client.get()
   local conky = { }
   local j = 1
   local i = 1
   while clients[i]
   do
      if clients[i].class == "conky"
      then
	 conky[j] = clients[i]
	 j = j + 1
      end
      i = i + 1
   end
   return conky
end
function raise_conky()
   local conky = get_conky()
   local i = 1
   while conky[i]
   do
      conky[i].ontop = true
      i = i + 1
   end
end
function lower_conky()
   local conky = get_conky()
   local i = 1
   while conky[i]
   do
      conky[i].ontop = false
      i = i + 1
   end
end
Wallpaper
local function set_wallpaper(s)
   -- Wallpaper
   if beautiful.wallpaper then
      local wallpaper = beautiful.wallpaper
      -- If wallpaper is a function, call it with the screen
      if type(wallpaper) == "function" then
	 wallpaper = wallpaper(s)
      end
      gears.wallpaper.maximized(wallpaper, s, true)
   end
end

  -- Re-set wallpaper when a screen's geometry changes (e.g. different resolution)
  screen.connect_signal("property::geometry", set_wallpaper)
Per-window keyboard layout
require("kbdlayout")

Credit: https://gist.github.com/necauqua/4faa6a2cba66435e23fa98578862a0c8

local layout = require 'awful.widget.keyboardlayout'
local menubar = require 'menubar'

local kbdlayout = {
    globally_preferred = 'us',
    menubar_preferred = 'us',
}

local function get_idx_by_name(name)
    if not name then
        return 
    end
    for i, v in ipairs(layout.get_groups_from_group_names(awesome.xkb_get_group_names())) do
        if v.file == name then
            return i - 1
        end
    end
end

require 'awful.client' .property.persist('last_layout', 'number')

local oneshot_lock = false

local function on_layout_change()
    -- so that it does not override the last_layout
    -- when we set it, e.g. from menubar.show
    if oneshot_lock then
        oneshot_lock = false
        return
    end
    local c = client.focus
    if c then
        local idx = awesome.xkb_get_layout_group()
        c.last_layout = idx
    end
end

local function on_focus_changed(c)
    local idx = c.last_layout or get_idx_by_name(c.preferred_layout or kbdlayout.globally_preferred)
    if idx and awesome.xkb_get_layout_group() ~= idx then
        awesome.xkb_set_layout_group(idx)
    end
end

awesome.connect_signal('xkb::map_changed', on_layout_change)
awesome.connect_signal('xkb::group_changed', on_layout_change)
client.connect_signal('focus', on_focus_changed)

-- menubar has no signals or anything, so just plain old monkeypatching

local menubar_show = menubar.show
local menubar_hide = menubar.hide

function menubar.show(...)
    menubar_show(...)
    local idx = get_idx_by_name(kbdlayout.menubar_preferred)
    if idx then
        oneshot_lock = true
        awesome.xkb_set_layout_group(idx)
    end
end

function menubar.hide(...)
    menubar_hide(...)
    local c = client.focus
    if c then
        on_focus_changed(c)
    end
end

return kbdlayout
Screen layouts
 awful.screen.connect_for_each_screen(
    function(s)
	-- Wallpaper
	set_wallpaper(s)

	-- Each screen has its own tag table.
	-- if s.index == 1 then
	awful.tag({ "emacs", "web", "tile", "float", "steam" }, s, { awful.layout.suit.tile.bottom, awful.layout.suit.tile.left, awful.layout.suit.spiral.dwindle, awful.layout.suit.floating, awful.layout.suit.floating })
	root.tags()[1].master_width_factor = 0.65
	root.tags()[1].useless_gap = 0
	-- else
	--    awful.tag({"present","data"}, s, awful.layout.layouts[1])
	-- end

	-- Create an imagebox widget which will contains an icon indicating which layout we're using.
	-- We need one layoutbox per screen.
	s.mylayoutbox = awful.widget.layoutbox(s)
	-- Create a taglist widget
	s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, taglist_buttons)

	-- Create a tasklist widget

	s.mytasklist_nofocus = awful.widget.tasklist(s, function (c,screen) return awful.widget.tasklist.filter.currenttags (c, screen) end, nil, nil, nil, wibox.layout.flex.horizontal())

	-- Create the wibox
	s.mywibox = awful.wibar({ position = "top", screen = s })
	s.mywibox_status = awful.wibar({ position = "bottom", screen = s })
	s.mywibox_right = awful.wibar({ position = "right", screen = s })

	-- Add widgets to the wibox
	s.mywibox:setup {
	   layout = wibox.layout.align.horizontal,
	   s.mylayoutbox,
	   s.mytasklist_nofocus,
	   { -- Right widgets
	      layout = wibox.layout.fixed.horizontal,
	      s.index == 1 and spacing,
	      s.index == 1 and balance,
	      arr1,
	      spacing
	   },
	}
	s.mywibox_right:setup {
	   layout = wibox.layout.align.vertical,
	   {
	      layout = wibox.layout.fixed.vertical,
	      s.index == 1 and spacing,
	      s.index == 1 and mysystray,
	   },
	   {
	      layout = wibox.layout.fixed.vertical,
	   },
	   {
	      s.mytaglist,
	      direction = 'west',
	      layout = wibox.container.rotate,
	   },
	}
	s.mywibox_status:setup {
	   layout = wibox.layout.align.horizontal,
	   expand = "none",
	   s.index == 1 and orgstatus,
	   {
	      layout =  wibox.layout.fixed.horizontal,
	      forced_width = 1,
	      spacing,
	   },
	   {
	      layout =  wibox.layout.fixed.horizontal,
	      cpuicon,
	      cpuwidget,
	      memicon,
	      memwidget,
	      volumeicon,
	      volume,
	      baticon,
	      batwidget,
	      neticon,
	      netwidget,
	      clockicon,
	      tdwidget,
	      -- vpnicon,
	      diagicon,
	   },
	}
 end)
 -- }}}
Key bindings
function initflagmove ()
   keynumber = 0
   -- Bind all key numbers to tags.
   -- Be careful: we use keycodes to make it works on any keyboard layout.
   -- This should map on the top row of your keyboard, usually 1 to 9.
   tagkeys={"x","z","s","d","f"}
   for s in screen do
	keynumber = math.min(9, math.max(#s.tags, keynumber));
   end

   for i = 1, keynumber do
	globalkeys = awful.util.table.join(globalkeys,
					   -- View tag only.
					   awful.key({ modkey, "Control" }, tagkeys[i],
					      function ()
							local screen = awful.screen.focused()
							local tag = screen.tags[i]
							if tag then
							   tag:view_only()
							end
						     end,
						     {description = "view tag #"..i, group = "tag"}),
					   -- Move client to tag.
					   awful.key({ modkey, "Shift" }, tagkeys[i],
					      function ()
							if client.focus then
							   local tag = client.focus.screen.tags[i]
							   if tag then
							      client.focus:move_to_tag(tag)
							   end
							end
						     end,
						     {description = "move focused client to tag #"..i, group = "tag"})
					   -- -- Toggle tag on focused client.
					   -- awful.key({ modkey, "Alt_L", "Shift" }, tagkeys[i],
					   -- 	   function ()
					   -- 	      if client.focus then
					   -- 		 local tag = client.focus.screen.tags[i]
					   -- 		 if tag then
					   -- 		    client.focus:toggle_tag(tag)
					   -- 		 end
					   -- 	      end
					   -- 	   end,
					   -- 	   {description = "toggle focused client on tag #" .. i, group = "tag"})
	)
   end
end

globalkeys = awful.util.table.join(
   awful.key({ modkey}, "j",
	function ()
		  awful.client.focus.byidx( 1)
		  if client.focus then client.focus:raise() end
	       end),
   awful.key({ modkey }, "k",
	function ()
		  awful.client.focus.byidx(-1)
		  if client.focus then client.focus:raise() end
	       end),
   awful.key({ modkey}, "w", function () awful.spawn("bash -c 'nice -n 17 show-menu'") end),
   -- Layout manipulation
   awful.key({ modkey, "Shift" }, "j", function () awful.client.swap.byidx( 1) awful.client.focus.byidx(-1) end),
   awful.key({ modkey, "Shift" }, "k", function () awful.client.swap.byidx(-1) awful.client.focus.byidx( 1) end),
   awful.key({ modkey,           }, "Tab",
	function (t)
		  awful.tag.history.restore(awful.tag.getscreen(t))
	       end),
   awful.key({ modkey,          }, "q",
	function (t)
		  awful.client.focus.history.previous()
		  if client.focus then
		     client.focus:raise()
		  end
	       end),
   -- Standard program
   awful.key({ modkey         }, "Return", function() awful.spawn("bash -c \"trackball-toggle.sh enable && xtrlock && trackball-toggle.sh disable\"") end),
   awful.key({ modkey},"r", function() awful.spawn("bash -c 'PATH=\"/home/yantar92/.local/bin:$PATH\" nice -n 17 emacs-helm-menu'") end),
   awful.key({ modkey},"y", function() awful.spawn("bash -c 'passmenu --type'") end),
   awful.key({ modkey},"u", function() awful.spawn("bash -c 'passmenu --type --user'") end),
   awful.key({ modkey}, "t", function () awful.spawn("bash -c 'run_single_window.sh \"Kitty\" \"kitty\"'") end),
   awful.key({ modkey}, "=", function () awful.spawn("show-dictionary.sh") end),
   awful.key({ modkey}, "-", function () awful.spawn("show-translation.sh") end),
   awful.key({ modkey, "Control" }, "c", function () awful.spawn("mingus-start.sh") end),
   awful.key({ modkey, "Control", "Shift" }, "c", function () awful.spawn("mingus-del-current.sh") end),
   -- awful.key({ modkey, "Control" }, "w", function () awful.spawn("elfeed-start.sh") end),
   awful.key({ modkey, "Control" }, "g", function () awful.spawn("notmuch-emacs-start.sh --inbox") end),
   awful.key({ modkey, "Control", "Shift" }, "g", function () awful.spawn("notmuch-emacs-start.sh --newsonly") end),
   awful.key({ modkey }, "e", function () awful.spawn("emacs-start.sh") end),
   awful.key({ modkey }, "i", function () awful.spawn("qutebrowser-start.sh") end),
   awful.key({ modkey }, "v", function () awful.spawn("clipmenu.sh") end),
   -- awful.key({ modkey, "Control" }, "u", function () awful.spawn("urxvt -e rtorrent-start.sh") end),
   -- awful.key({ modkey, "Control" }, "b", function () awful.spawn("urxvt -e start_bc") end),
   awful.key({ modkey, "Shift" }, "p", function () awful.spawn("mpd-toggle.sh") end),
   -- awful.key({ modkey, "Shift" }, "=", function () awful.spawn("mpc -q volume +5") end),
   -- awful.key({ modkey, "Shift" }, "-", function () awful.spawn("mpc -q volume -5") end),
   awful.key({ modkey, "Shift" }, ".", function () awful.spawn("mpd-next.sh") end),
   awful.key({ modkey, "Shift" }, ",", function () awful.spawn("mpd-prev.sh") end),
   awful.key({ }, "XF86AudioNext", function () awful.spawn("mpd-next.sh") end),
   awful.key({ }, "XF86AudioPrev", function () awful.spawn("mpd-prev.sh") end),
   awful.key({ }, "XF86AudioPlay", function () awful.spawn("mpd-toggle.sh") end),
   -- awful.key({ }, "XF86Tools", function () awful.spawn("vpn-select-server.sh") end),
   -- awful.key({ }, "F9", function () awful.spawn("vpn-toggle.sh") end),
   -- awful.key({ modkey, "Control" }, "q", function () awful.spawn("equalizer-start.sh") end),
   awful.key({modkey, "Control", "Shift" }, "o",     function () awful.screen.focus_relative(1) end),
   --   awful.key({ "Control" }, "1", function () awful.spawn("seltr") end),
   awful.key({ }, "XF86AudioMute", function () awful.spawn("mute.sh", false) end),
   awful.key({ }, "XF86AudioLowerVolume", function () awful.spawn("dec_volume.sh", false) end),
   awful.key({ }, "XF86AudioRaiseVolume", function () awful.spawn("inc_volume.sh", false) end),
   awful.key({ }, "XF86Favorites", function () awful.spawn("trackball-toggle.sh") end),
   -- awful.key({ }, "Print", function () awful.spawn("sh -c '(cd ~/ && import -window root screenshot.jpg && mv screenshot.jpg screenshot.`date +%d.%m.%Y-%H.%M.%S`.jpg)'") end),
   awful.key({ }, "Print", function () awful.spawn("sh -c 'flameshot gui'") end),
   awful.key({ modkey, "Shift" }, "i", function () awful.spawn("org-clock-in-organization.sh", false) end),
   awful.key({ modkey, "Shift" }, "5", function () awful.spawn("org-clock-in-bookmarks.sh", false) end),
   awful.key({ modkey, "Shift" }, "6", function () awful.spawn("org-clock-out.sh", false) end),
   awful.key({ modkey }, "c", function () awful.spawn("bash -c 'emacsclient -e \"(org-capture)\"'", false) end),
   -- Tilewidth manupulation
   awful.key({ modkey, "Shift"}, "l",     function () awful.tag.incmwfact( 0.05)    end),
   awful.key({ modkey, "Shift"}, "h",     function () awful.tag.incmwfact(-0.05)    end),
   awful.key({ modkey, "Control", "Shift" }, "h",     function () awful.tag.incncol( 1)         end),
   awful.key({ modkey, "Control", "Shift" }, "l",     function () awful.tag.incncol(-1)         end),
   awful.key({ modkey,           }, "space", function () awful.layout.inc(awful.layout.layouts,  1) end),
   awful.key({ modkey, "Shift"   }, "space", function () awful.layout.inc(awful.layout.layouts, -1) end),
   awful.key({ modkey}, "BackSpace", function ()
		  for s in pairs(naughty.notifications) do
		     for p in pairs(naughty.notifications[s]) do
			for i,notification in pairs(naughty.notifications[s][p]) do
			   if notification then
			      naughty.destroy(notification)
			      return notification
			   end
			end
		     end
		  end
				       end),
   awful.key({ modkey, "Control", "Shift" }, "n", function()
		  c=awful.client.restore()
		  if c==nil then return end
		  client.focus=c
		  if client.focus then
		     client.focus:raise()
		  end
						    end),
   -- Prompt
   awful.key({}, "F11", function() raise_conky() end, function() lower_conky() end)
   -- awful.key({ modkey, "Shift", "Control" }, "n",
   -- 	     function()
   -- 		local tag = awful.tag.selected()
   -- 		for i=1, #tag:clients() do
   -- 		   tag:clients()[i].minimized=false
   -- 		end
   -- 	     end),
   -- awful.key({ modkey, "Shift"       }, "n",
   -- 	       function()
   -- 		  -- client_list={}
   -- 		  -- local tag = awful.tag.selected()
   -- 		  -- for i=1, #tag:clients() do
   -- 		  --    cl=tag:clients()[i]
   -- 		  --    if tag:clients()[i].minimized then
   -- 		  --       prefix = "_ "
   -- 		  --    else
   -- 		  --       prefix = "* "
   -- 		  --    end
   -- 		  --    if not awful.rules.match(cl, {class= "conky"}) then
   -- 		  --       client_list[i]=
   -- 		  -- 	 {prefix .. cl.name.."</span>",
   -- 		  -- 	  function()
   -- 		  -- 	     cl_menu:hide()
   -- 		  -- 	     tag:clients()[i].minimized=not tag:clients()[i].minimized
   -- 		  -- 	  end,
   -- 		  -- 	  cl.icon
   -- 		  -- 	 }
   -- 		  --    end
   -- 		  -- end
   -- 		  -- cl_menu=awful.menu({items = client_list, theme = {width=300}})
   -- 		  -- mouse.coords({x=533,y=354})
   -- 		  -- cl_menu:show({keygrabber = true})
   -- 		  awful.spawn("windowcd.sh")
   -- 	       end
   -- )
)

clientkeys = awful.util.table.join(
   awful.key({ modkey, "Control"   }, "m",      function (c) c.fullscreen = not c.fullscreen  end),
   awful.key({ modkey, "Shift"   }, "'",      function (c) c:kill()                         end),
   awful.key({ modkey, "Shift"   }, "q",      function (c)
		  for _,c2 in pairs(c.screen.clients) do
		     if c2 ~= c then
			c2:kill()
		     end
		  end
						end),
   awful.key({ modkey, "Control" }, "space",  awful.client.floating.toggle                     ),
   awful.key({ modkey, "Shift"   }, "o",      awful.client.movetoscreen                        ),
   awful.key({ modkey, "Control", "Shift" }, "t",      function (c) c.ontop = not c.ontop            end),
   -- awful.key({ modkey,           }, "u", function (c) c.marked = not c.marked end ),
   awful.key({ modkey, "Control"}, "n",
	       function (c)
		  c.minimized = true
	       end),
   awful.key({ modkey }, "m",
	       function (c)
		  c.maximized_horizontal = not c.maximized_horizontal
		  c.maximized_vertical   = not c.maximized_vertical
	       end)
)


initflagmove()

clientbuttons = awful.util.table.join(
   --    awful.button({ }, 1, function (c) client.focus = c; c:raise() end),
   awful.button({ modkey }, 1, function(c) awful.mouse.client.move() end),
   awful.button({ modkey }, 3, function(c) awful.mouse.client.resize() end))

-- Set keys
root.keys(globalkeys)
-- }}}
Client rules
 -- Rules to apply to new clients (through the "manage" signal).
 awful.rules.rules = {
	<<awesome-default-client-rule>>,
	<<awesome-default-titlebar-rule>>,
	<<awesome-default-dialog-rule>>,
	<<awesome-default-emacs-rule>>,
	<<awesome-default-pinentry-rule>>,
	<<awesome-default-keyring-rule>>,
	<<awesome-default-kitty-rule>>,
	<<awesome-default-gnuplot-rule>>,
	<<awesome-default-conky-rule>>,
	<<awesome-default-mpv-rule>>,
	<<awesome-default-astrill-rule>>,
	<<awesome-default-crqt-rule>>,
 }
Default
-- All clients will match this rule.
{ rule = { },
  properties = { border_width = beautiful.border_width,
		      border_color = beautiful.border_normal,
		      focus = awful.client.focus.filter,
		      raise = true,
		      keys = clientkeys,
		      buttons = clientbuttons,
		      screen = awful.screen.preferred,
		      placement = awful.placement.no_overlap+awful.placement.no_offscreen
  }
}
Clients with titlebar
-- Add titlebars to normal clients and dialogs
{ rule_any = { type = { "normal", "dialog" } },
  properties = { titlebars_enabled = true }
}
Dialogues
  { rule = { type = "dialog" },
    properties = {
	  floating = true
    }
  }
Emacs
  { rule = { class = "Emacs", name = "Helm frame"},
    properties = {
	  floating = true,
       placement = awful.placement.centered,
       skip_taskbar = true,
       ontop = true,
       maximized_horizontal = true,
       height = 800,
	  titlebars_enabled = false,
	  border_width = 1.0,
	  opacity = 0.9,
    },
    callback = function(c)
	  gears.apply_shape_bounding(c, gears.shape.rounded_rect, 15)
    end,
  }
Qutebrowser
  { rule = { class = "qutebrowser" },
    properties = {
	  tag = "web",
    },
  }
Pinentry prompt
  { rule = { class = "Pinentry" },
    properties = {
	  floating = true,
       placement = awful.placement.centered
    },
  }
Keyring prompt
  { rule = { class = "gcr-prompter" },
    properties = {
	  floating = true,
       placement = awful.placement.centered
    },
  }
Kitty
{ rule = {class = "kitty"},
       properties = {
	  floating = true,
	  skip_taskbar = true,
	  titlebars_enabled = false,
	  width = 1878,
	  height = 500,
	  border_width = 0
       }
     }
Gnuplot
  { rule = {class = "gnuplot_qt"},
    properties = {
	  floating = true,
	  above = true,
    }
  }
Conky
  { rule = { class = "conky" },
    properties = {
	  floating = true,
	  sticky = true,
	  ontop = false,
	  focusable = false,
	  border_width = 0,
    }
  }
MPV
  { rule = { class = "mpv" },
    properties = {
	  floating = true
    }
  }
Astrill
  { rule = { class = "Astrill" },
    properties = {
	  floating = true,
	  placement = awful.placement.top_right
    }
  }
Cool reader
  { rule = { class = "crqt" },
    properties = {
	  fullscreen = true
    }
  }
Client signals
-- {{{ Signals
-- Signal function to execute when a new client appears.
client.connect_signal("manage", function (c)
			   -- Set the windows at the slave,
			   -- i.e. put it at the end of others instead of setting it master.
			   -- if not awesome.startup then awful.client.setslave(c) end
			   if awesome.startup and
			      not c.size_hints.user_position
			      and not c.size_hints.program_position
			   then
			      -- Prevent clients from being unreachable after screen count changes.
			      awful.placement.no_offscreen(c)
			   end
				  end)

-- Add a titlebar if titlebars_enabled is set to true in the rules.
client.connect_signal("request::titlebars", function(c)
			   -- buttons for the titlebar
			   awful.titlebar(c, {position="bottom"}) : setup {
			      { -- Left
				 awful.titlebar.widget.iconwidget(c),
				 spacing,
				 buttons = buttons,
				 layout  = wibox.layout.fixed.horizontal
			      },
			      { -- Middle
				 { -- Title
				    align  = "left",
				    widget = awful.titlebar.widget.titlewidget(c)
				 },
				 buttons = buttons,
				 layout  = wibox.layout.flex.horizontal
			      },
			      spacing,
			      layout = wibox.layout.align.horizontal
									  }
					      end)

-- Enable sloppy focus, so that focus follows mouse.
-- client.connect_signal("mouse::enter", function(c)
--     if awful.layout.get(c.screen) ~= awful.layout.suit.magnifier
--         and awful.client.focus.filter(c) then
--         client.focus = c
--     end
-- end)


client.connect_signal("focus", function(c) c.border_color = beautiful.border_focus; if not c.class == "mpv" then c.above = true; end end)

client.connect_signal("unfocus", function(c) c.border_color = beautiful.border_normal; c.above=false; end)
-- }}}
Autostart \ startup
awful.spawn.with_shell("[[ -e /tmp/startup.run ]] || (mkdir -p ~/.log; ~/.config/awesome/startup >> ~/.log/awesome.log 2>&1) && touch /tmp/startup.run")
Helper scripts
Make sure that app is going to run in a single window

Requires =Wmctrl= and xprop

  function help() {
      cat <<EOF
   This scripts runs a program in a single window.
  If the program is already running, it raise the focus of its window.
  USAGE: start-single-window.sh [--toggle] "program unique id without special chars and spaces" "command to run"
  --toggle: hide the command window if in focus and show it otherwise
EOF
  }

  function wmctrl_getwindowname() {
      echo "$(xprop -id $1 | grep "^WM_NAME(" | cut -d\" -f2)"
  }

  TOGGLE=""
  if [[ "$1" == "--toggle" ]]; then
      TOGGLE="1"
      shift;
  fi

  [[ "$#" -ne "2" ]] && echo "Wrong number of parameters" && help && exit

  NAME="$1"
  WINDOW_CMD="$2"

  INFO_FOLDER="/tmp"

  [[ -e "$INFO_FOLDER/$NAME" ]] && WINDOW_ID=`cat $INFO_FOLDER/$NAME` || WINDOW_ID=""
  if [ -z "$WINDOW_ID" ]
  then
      rm -f $INFO_FOLDER/$NAME
  else
      [[ -z "`wmctrl -l | grep $WINDOW_ID`" ]] && rm $INFO_FOLDER/$NAME
  fi

  if [ -e "$INFO_FOLDER/$NAME" ]
  then
      [[ "$(xdotool search --class "$NAME" | tail -n1)" =~ "$(xdotool getwindowfocus)" ]] && xdotool windowminimize $(xdotool getwindowfocus) || (wmctrl -iR "$WINDOW_ID"; wmctrl -iR "$WINDOW_ID";)
  else
      NUM_WINDOWS="`wmctrl -l | wc -l`"
      $WINDOW_CMD &
      JOB_ID="`jobs -p`"
      [[ -z "`ps ax | grep $JOB_ID`" ]] && echo "Process ID not found... stopping" && exit
      echo " Job id = $JOB_ID"
      FLAG="1"
      while [[ "$FLAG" -eq "1" ]];
      do
	  notify-send -t 450 "Running" "Running $NAME..."
	  while read WINDOW_ID;
	  do
	      echo Testing $WINDOW_ID ... $JOB_ID =? `xprop -id $WINDOW_ID | grep _NET_WM_PID | cut -d\  -f3`
	      if [ "$JOB_ID" -eq "`xprop -id "$WINDOW_ID" | grep _NET_WM_PID | cut -d\  -f3`" ]
	      then
		  echo "$WINDOW_ID" > $INFO_FOLDER/$NAME
		  echo "done... $NAME: window id = $WINDOW_ID"
		  FLAG="0"
		  break
	      fi
	      [[ "$FLAG" -eq "0" ]] && break
	  done <<< "`wmctrl -l | cut -d\  -f1`"
	  perl -e 'select(undef,undef,undef,.5)'
      done
      wmctrl -ir "`cat $INFO_FOLDER/$NAME`"
  fi

Run/search menu via Emacs helm

#launcher #runner Inspired by [[id:Mattduckemacs_as_fuzzy_launc_and_alfred_replac6c6][[Mattduck] Emacs as a fuzzy launcher and Alfred-replacement]]
if [[ "$(getactvwindclass)" == "emacs.Emacs" ]]; then
    # Relies on my key bindings
    xdotool key --clearmodifiers --delay 60 alt+l e h;
    # release any pressed modifiers
    xdotool keyup super alt ctrl shift
else
    emacsclient -c --frame-parameters="((title . \"Helm frame\") (unsplittable . t))" -e "(let ((inhibit-message t) (helm-full-frame t) (helm-use-undecorated-frame-option nil)) (catch :quit (yant/helm-org-ql-agenda-files nil) (when (= 0 helm-exit-status) (let ((fr (selected-frame))) (make-frame) (delete-frame fr))) (unless (= 0 helm-exit-status) (user-error \"%s\" \"Exit frame\"))))";
fi

Credit: https://github.com/computefoundation/gnu-linux-shell-scripting/blob/master/scripts/x11_management-output_only/window_property_information/getactvwindclass (with fixes)

# 
# File:
#   getactvwindclass
# 
# Description:
#   Get the class of the active window.
# 
# Returns:
#   Full qualified class string (class and classname; e.g. "xterm.XTerm")) of
#   window
# 

getActvWindId() {
  echo "$(xprop -root _NET_ACTIVE_WINDOW | cut -d ' ' -f 5)"
}

getWindClassById() {
  echo "$(xprop -id "${1}" WM_CLASS | sed -n \
      's/[^"]*"\([^"]*\)",\s"\([^"]*\)"/\1.\2/p')"
}

echo "$(getWindClassById "$(getActvWindId)")"

Tools

Rofi quick menu

x11-misc/rofi
Configuration
configuration {
        separator-style: "solid";
        scrollbar-width: 5;
        terminal: "urxvt";
        width: 35;
        lines: 7;
        padding: 10;
        run-command: "bash -c '{cmd}'";
        window-command: "wmctrl -b toggle,hidden -i -r {window}";
        font: "Adobe Garamond 15";
        kb-row-down: "Down,Control+n,Alt+j";
        kb-row-up: "Up,Control+p,Shift+Tab,Shift+ISO_Left_Tab,Alt+k";
}
@theme "/usr/share/rofi/themes/Arc-Dark.rasi"
Menu generator for rofi: menugen
wite ebuildEND

../Dist/menugen-master/

Personal menu
Main menu
#begin setup
PATH="$PATH:/home/yantar92/.local/bin"
ROFI_OPTS="-width 20 -lines 20 -auto-select"
#end setup

#begin main
name="Main menu"
add_item	'Pendrives list >'	'next'		'show-flash.sh'
add_link 'System tools>'            'system'
add_link 'Awesome menu>'            'awesome'
#end main

#begin system
name="System"
add_exec	'Audio Control'	'pavucontrol-qt'
add_exec	'Video Control'	'qv4l2'
add_exec	'Wifi Networks'	'sudo wpa_gui'
#end system

#begin awesome
name="Awesome"
add_exec	'Manual'		"xdg-open /usr/share/doc/awesome-4.2-r3/doc/index.html"
add_exec	'Restart Awesome'	"killall xxkb; conky-stop; echo 'awesome.restart()' | awesome-client"
add_exec	'Quit Awesome'	"killall xxkb; conky-stop; echo 'awesome.quit()' | awesome-client"
#end awesome
Pendrive menu
menu-flash.sh > /tmp/1.sh && menugen /tmp/1.sh
if [ $# = 1 ]; then
    echo "#begin $1"
    [ -z "`mount | grep "$1"`" ] && echo "name='$1(not mounted)'" || echo "name='$1'"
    if [ ! -z "`mount | grep "$1"`" ]; 
    then
	echo "add_exec 'Open' 'xdg-open /mnt/$1'"
	echo "add_exec 'Umount' \"sh -c 'umount+pop.sh $1'\""
    else
	#	[ "$1" = "Samba" ] && echo "prog 'Mount' - sh -c 'fusesmb.mount && notify-send Samba \" Successfully mounted\"'" && exit
	#	[ "$1" = "cdrom" ] && echo "prog 'Mount' - sh -c 'sudo mount /dev/cdrom /mnt/cdrom && notify-send Cdrom \"Successfully mounted\"'" && exit
	#   [ -d "/mnt/$1" ] && echo "prog 'Mount' - sh -c 'sudo \"`ls -l /mnt/$1/ | cut -d' ' -f 12 | tail -n1` -o utf8,gid=100,umask=002 /dev/`ls /mnt/$1/`\" /mnt/$1 && /bin/touch /tmp/$1'"
	[ -d "/mnt/$1" ] && echo "add_exec 'Mount' \"sudo `/bin/ls -l /mnt/$1 | awk '{print $11}' | tail -n1` -O utf8,gid=100,umask=002 /dev/`/bin/ls /mnt/$1` /mnt/$1 && /bin/touch /tmp/$1 && mount-pop.sh \""
    fi
    echo "#end $1"
else
    Folders="`find /mnt/ -maxdepth 1 -mindepth 1 -type d -printf "%f\n"`"
    ZZ="0"
    echo "#\!menugen"
    echo "#begin setup"
    echo "PATH=\"$PATH:/home/yantar92/.local/bin\""
    echo "ROFI_OPTS=\"-width 30 -lines 20 -auto-select -hide-scrollbar\""
    echo "#end setup"
    echo "#begin main"
    for x in $Folders; do
	echo "add_link '$x' '$x'"
	ZZ="1"
    done
    if [[ "$ZZ" = "0" ]]; then
	echo "name=\"No pendrives detected\""
	echo "numbered=false"
	echo "add_exec ' ' 'true'"
	echo "#end main"
    else
	echo "name=\"Pendrives\""
	echo "#end main"
	for x in $Folders; do
	    menu-flash.sh "$x"
	done
    fi
fi

Notification script

The scripts are relying on the way USB automount is implemented.

sudo umount /mnt/$1 > /tmp/umount.log 2>&1 && notify-send "$1" "Successfully unmounted" || notify-send -u critical "$1" "`cat /tmp/umount.log`" || rm /tmp/umount.log
DISPLAY=:0
# export DBUS_SESSION_BUS_ADDRESS environment variable
# PID=$(pgrep awesome | head -n1)
# [[ -z "$PID" && -d "/proc/$PID" ]] && export DBUS_SESSION_BUS_ADDRESS="$(grep -z DBUS_SESSION_BUS_ADDRESS /proc/$PID/environ | tr '\0' '\n' | cut -d= -f2-)"

Folders="`find /mnt/ -maxdepth 1 -mindepth 1 -type d -printf "%f\n"`"
for x in $Folders; do
    [ -f "/tmp/$x" ] && notify-send -u critical -t 5000 "$x" "Successfully mounted" && rm "/tmp/$x";
done

Sound control: pavucontrol-qt

media-sound/pavucontrol-qt

Read key evens (xev)

x11-apps/xev

net-misc/ssh-askpass-fullscreen

Required for straight.el in emacs
net-misc/ssh-askpass-fullscreen

Screenshot (flameshot)

media-gfx/flameshot

Other screenshot utility: scrot + gifsicle to compress gifs

Terminal: Kitty

Installation

x11-terms/kitty

Use the cutting edge version: [2021-04-10 Sat] Do not use it, Gentoo ebuild is not working for master now.

x11-terms/kitty ~amd64
x11-terms/kitty-terminfo ~amd64

Required use changes:

media-libs/libcanberra alsa

Configuration

Appearance
# vim:fileencoding=utf-8:ft=conf:foldmethod=marker

#: Fonts {{{

#: kitty has very powerful font management. You can configure
#: individual font faces and even specify special fonts for particular
#: characters.

font_family      monospace
bold_font        auto
italic_font      auto
bold_italic_font auto

#: You can specify different fonts for the bold/italic/bold-italic
#: variants. To get a full list of supported fonts use the `kitty
#: list-fonts` command. By default they are derived automatically, by
#: the OSes font system. Setting them manually is useful for font
#: families that have many weight variants like Book, Medium, Thick,
#: etc. For example::

#:     font_family      Operator Mono Book
#:     bold_font        Operator Mono Medium
#:     italic_font      Operator Mono Book Italic
#:     bold_italic_font Operator Mono Medium Italic

#font_size 11.0
font_size 15

#: Font size (in pts)

adjust_line_height  0
adjust_column_width 0

#: Change the size of each character cell kitty renders. You can use
#: either numbers, which are interpreted as pixels or percentages
#: (number followed by %), which are interpreted as percentages of the
#: unmodified values. You can use negative pixels or percentages less
#: than 100% to reduce sizes (but this might cause rendering
#: artifacts).

# symbol_map U+E0A0-U+E0A2,U+E0B0-U+E0B3 PowerlineSymbols

#: Map the specified unicode codepoints to a particular font. Useful
#: if you need special rendering for some symbols, such as for
#: Powerline. Avoids the need for patched fonts. Each unicode code
#: point is specified in the form U+<code point in hexadecimal>. You
#: can specify multiple code points, separated by commas and ranges
#: separated by hyphens. symbol_map itself can be specified multiple
#: times. Syntax is::

#:     symbol_map codepoints Font Family Name

disable_ligatures never

#: Choose how you want to handle multi-character ligatures. The
#: default is to always render them.  You can tell kitty to not render
#: them when the cursor is over them by using cursor to make editing
#: easier, or have kitty never render them at all by using always, if
#: you don't like them. The ligature strategy can be set per-window
#: either using the kitty remote control facility or by defining
#: shortcuts for it in kitty.conf, for example::

#:     map alt+1 disable_ligatures_in active always
#:     map alt+2 disable_ligatures_in all never
#:     map alt+3 disable_ligatures_in tab cursor

box_drawing_scale 0.001, 1, 1.5, 2

#: Change the sizes of the lines used for the box drawing unicode
#: characters These values are in pts. They will be scaled by the
#: monitor DPI to arrive at a pixel value. There must be four values
#: corresponding to thin, normal, thick, and very thick lines.

#: }}}

#: Cursor customization {{{

cursor #cccccc

#: Default cursor color

cursor_text_color #111111

#: Choose the color of text under the cursor. If you want it rendered
#: with the background color of the cell underneath instead, use the
#: special keyword: background

cursor_shape block

#: The cursor shape can be one of (block, beam, underline)

cursor_blink_interval -1

#: The interval (in seconds) at which to blink the cursor. Set to zero
#: to disable blinking. Negative values mean use system default. Note
#: that numbers smaller than repaint_delay will be limited to
#: repaint_delay.

cursor_stop_blinking_after 15.0

#: Stop blinking cursor after the specified number of seconds of
#: keyboard inactivity.  Set to zero to never stop blinking.

#: }}}
Text buffer
#: Scrollback {{{

scrollback_lines 2000

#: Number of lines of history to keep in memory for scrolling back.
#: Memory is allocated on demand. Negative numbers are (effectively)
#: infinite scrollback. Note that using very large scrollback is not
#: recommended as it can slow down resizing of the terminal and also
#: use large amounts of RAM.

scrollback_pager less --chop-long-lines --RAW-CONTROL-CHARS +INPUT_LINE_NUMBER

#: Program with which to view scrollback in a new window. The
#: scrollback buffer is passed as STDIN to this program. If you change
#: it, make sure the program you use can handle ANSI escape sequences
#: for colors and text formatting. INPUT_LINE_NUMBER in the command
#: line above will be replaced by an integer representing which line
#: should be at the top of the screen.

scrollback_pager_history_size 0

#: Separate scrollback history size, used only for browsing the
#: scrollback buffer (in MB). This separate buffer is not available
#: for interactive scrolling but will be piped to the pager program
#: when viewing scrollback buffer in a separate window. The current
#: implementation stores one character in 4 bytes, so approximatively
#: 2500 lines per megabyte at 100 chars per line. A value of zero or
#: less disables this feature. The maximum allowed size is 4GB.

wheel_scroll_multiplier 5.0

#: Modify the amount scrolled by the mouse wheel. Note this is only
#: used for low precision scrolling devices, not for high precision
#: scrolling on platforms such as macOS and Wayland. Use negative
#: numbers to change scroll direction.

touch_scroll_multiplier 1.0

#: Modify the amount scrolled by a touchpad. Note this is only used
#: for high precision scrolling devices on platforms such as macOS and
#: Wayland. Use negative numbers to change scroll direction.

#: }}}
Mouse
#: Mouse {{{

mouse_hide_wait 3.0

#: Hide mouse cursor after the specified number of seconds of the
#: mouse not being used. Set to zero to disable mouse cursor hiding.
#: Set to a negative value to hide the mouse cursor immediately when
#: typing text.

url_color #0087bd
url_style curly

#: The color and style for highlighting URLs on mouse-over. url_style
#: can be one of: none, single, double, curly

open_url_modifiers kitty_mod

#: The modifier keys to press when clicking with the mouse on URLs to
#: open the URL

open_url_with xdg-open

#: The program with which to open URLs that are clicked on. The
#: special value default means to use the operating system's default
#: URL handler.

copy_on_select no

#: Copy to clipboard or a private buffer on select. With this set to
#: clipboard, simply selecting text with the mouse will cause the text
#: to be copied to clipboard. Useful on platforms such as macOS that
#: do not have the concept of primary selections. You can instead
#: specify a name such as a1 to copy to a private kitty buffer
#: instead. Map a shortcut with the paste_from_buffer action to paste
#: from this private buffer. For example::

#:     map cmd+shift+v paste_from_buffer a1

#: Note that copying to the clipboard is a security risk, as all
#: programs, including websites open in your browser can read the
#: contents of the system clipboard.

strip_trailing_spaces never

#: Remove spaces at the end of lines when copying to clipboard. A
#: value of smart will do it when using normal selections, but not
#: rectangle selections. always will always do it.

rectangle_select_modifiers ctrl+alt

#: The modifiers to use rectangular selection (i.e. to select text in
#: a rectangular block with the mouse)

terminal_select_modifiers shift

#: The modifiers to override mouse selection even when a terminal
#: application has grabbed the mouse

select_by_word_characters :@-./_~?&=%+#

#: Characters considered part of a word when double clicking. In
#: addition to these characters any character that is marked as an
#: alpha-numeric character in the unicode database will be matched.

click_interval -1.0

#: The interval between successive clicks to detect double/triple
#: clicks (in seconds). Negative numbers will use the system default
#: instead, if available, or fallback to 0.5.

focus_follows_mouse no

#: Set the active window to the window under the mouse when moving the
#: mouse around

pointer_shape_when_grabbed arrow

#: The shape of the mouse pointer when the program running in the
#: terminal grabs the mouse.

#: }}}
Performance tuning
#: Performance tuning {{{

repaint_delay 10

#: Delay (in milliseconds) between screen updates. Decreasing it,
#: increases frames-per-second (FPS) at the cost of more CPU usage.
#: The default value yields ~100 FPS which is more than sufficient for
#: most uses. Note that to actually achieve 100 FPS you have to either
#: set sync_to_monitor to no or use a monitor with a high refresh
#: rate. Also, to minimize latency when there is pending input to be
#: processed, repaint_delay is ignored.

input_delay 3

#: Delay (in milliseconds) before input from the program running in
#: the terminal is processed. Note that decreasing it will increase
#: responsiveness, but also increase CPU usage and might cause flicker
#: in full screen programs that redraw the entire screen on each loop,
#: because kitty is so fast that partial screen updates will be drawn.

sync_to_monitor yes

#: Sync screen updates to the refresh rate of the monitor. This
#: prevents tearing (https://en.wikipedia.org/wiki/Screen_tearing)
#: when scrolling. However, it limits the rendering speed to the
#: refresh rate of your monitor. With a very high speed mouse/high
#: keyboard repeat rate, you may notice some slight input latency. If
#: so, set this to no.

#: }}}
Terminal bell
#: Terminal bell {{{

enable_audio_bell no

#: Enable/disable the audio bell. Useful in environments that require
#: silence.

visual_bell_duration 0.0

#: Visual bell duration. Flash the screen when a bell occurs for the
#: specified number of seconds. Set to zero to disable.

window_alert_on_bell yes

#: Request window attention on bell. Makes the dock icon bounce on
#: macOS or the taskbar flash on linux.

bell_on_tab yes

#: Show a bell symbol on the tab if a bell occurs in one of the
#: windows in the tab and the window is not the currently focused
#: window

command_on_bell none

#: Program to run when a bell occurs.

#: }}}
Window layout
#: Window layout {{{

remember_window_size  yes
initial_window_width  640
initial_window_height 400

#: If enabled, the window size will be remembered so that new
#: instances of kitty will have the same size as the previous
#: instance. If disabled, the window will initially have size
#: configured by initial_window_width/height, in pixels. You can use a
#: suffix of "c" on the width/height values to have them interpreted
#: as number of cells instead of pixels.

enabled_layouts *

#: The enabled window layouts. A comma separated list of layout names.
#: The special value all means all layouts. The first listed layout
#: will be used as the startup layout. For a list of available
#: layouts, see the
#: https://sw.kovidgoyal.net/kitty/index.html#layouts.

window_resize_step_cells 2
window_resize_step_lines 2

#: The step size (in units of cell width/cell height) to use when
#: resizing windows. The cells value is used for horizontal resizing
#: and the lines value for vertical resizing.

window_border_width 1.0

#: The width (in pts) of window borders. Will be rounded to the
#: nearest number of pixels based on screen resolution. Note that
#: borders are displayed only when more than one window is visible.
#: They are meant to separate multiple windows.

draw_minimal_borders yes

#: Draw only the minimum borders needed. This means that only the
#: minimum needed borders for inactive windows are drawn. That is only
#: the borders that separate the inactive window from a neighbor. Note
#: that setting a non-zero window margin overrides this and causes all
#: borders to be drawn.

window_margin_width 0.0

#: The window margin (in pts) (blank area outside the border)

single_window_margin_width -1000.0

#: The window margin (in pts) to use when only a single window is
#: visible. Negative values will cause the value of
#: window_margin_width to be used instead.

window_padding_width 0.0

#: The window padding (in pts) (blank area between the text and the
#: window border)

placement_strategy center

#: When the window size is not an exact multiple of the cell size, the
#: cell area of the terminal window will have some extra padding on
#: the sides. You can control how that padding is distributed with
#: this option. Using a value of center means the cell area will be
#: placed centrally. A value of top-left means the padding will be on
#: only the bottom and right edges.

active_border_color #00ff00

#: The color for the border of the active window. Set this to none to
#: not draw borders around the active window.

inactive_border_color #cccccc

#: The color for the border of inactive windows

bell_border_color #ff5a00

#: The color for the border of inactive windows in which a bell has
#: occurred

inactive_text_alpha 1.0

#: Fade the text in inactive windows by the specified amount (a number
#: between zero and one, with zero being fully faded).

hide_window_decorations no

#: Hide the window decorations (title-bar and window borders). Whether
#: this works and exactly what effect it has depends on the window
#: manager/operating system.

resize_debounce_time 0.1

#: The time (in seconds) to wait before redrawing the screen when a
#: resize event is received. On platforms such as macOS, where the
#: operating system sends events corresponding to the start and end of
#: a resize, this number is ignored.

resize_draw_strategy static

#: Choose how kitty draws a window while a resize is in progress. A
#: value of static means draw the current window contents, mostly
#: unchanged. A value of scale means draw the current window contents
#: scaled. A value of blank means draw a blank window. A value of size
#: means show the window size in cells.

#: }}}
Tab bar
#: Tab bar {{{

tab_bar_edge top

#: Which edge to show the tab bar on, top or bottom

tab_bar_margin_width 0.0

#: The margin to the left and right of the tab bar (in pts)

tab_bar_style fade

#: The tab bar style, can be one of: fade, separator or hidden. In the
#: fade style, each tab's edges fade into the background color, in the
#: separator style, tabs are separated by a configurable separator.

tab_bar_min_tabs 2

#: The minimum number of tabs that must exist before the tab bar is
#: shown

tab_switch_strategy previous

#: The algorithm to use when switching to a tab when the current tab
#: is closed. The default of previous will switch to the last used
#: tab. A value of left will switch to the tab to the left of the
#: closed tab. A value of last will switch to the right-most tab.

tab_fade 0.25 0.5 0.75 1

#: Control how each tab fades into the background when using fade for
#: the tab_bar_style. Each number is an alpha (between zero and one)
#: that controls how much the corresponding cell fades into the
#: background, with zero being no fade and one being full fade. You
#: can change the number of cells used by adding/removing entries to
#: this list.

tab_separator " ┇"

#: The separator between tabs in the tab bar when using separator as
#: the tab_bar_style.

tab_title_template {title}

#: A template to render the tab title. The default just renders the
#: title. If you wish to include the tab-index as well, use something
#: like: {index}: {title}. Useful if you have shortcuts mapped for
#: goto_tab N.

active_tab_foreground   #000
active_tab_background   #eee
active_tab_font_style   bold-italic
inactive_tab_foreground #444
inactive_tab_background #999
inactive_tab_font_style normal

#: Tab bar colors and styles

#: }}}
Color scheme
#: Color scheme {{{

foreground #dddddd
background #000000

#: The foreground and background colors

background_opacity 1.0

#: The opacity of the background. A number between 0 and 1, where 1 is
#: opaque and 0 is fully transparent.  This will only work if
#: supported by the OS (for instance, when using a compositor under
#: X11). Note that it only sets the default background color's
#: opacity. This is so that things like the status bar in vim,
#: powerline prompts, etc. still look good.  But it means that if you
#: use a color theme with a background color in your editor, it will
#: not be rendered as transparent.  Instead you should change the
#: default background color in your kitty config and not use a
#: background color in the editor color scheme. Or use the escape
#: codes to set the terminals default colors in a shell script to
#: launch your editor.  Be aware that using a value less than 1.0 is a
#: (possibly significant) performance hit.  If you want to dynamically
#: change transparency of windows set dynamic_background_opacity to
#: yes (this is off by default as it has a performance cost)

dynamic_background_opacity no

#: Allow changing of the background_opacity dynamically, using either
#: keyboard shortcuts (increase_background_opacity and
#: decrease_background_opacity) or the remote control facility.

dim_opacity 0.75

#: How much to dim text that has the DIM/FAINT attribute set. One
#: means no dimming and zero means fully dimmed (i.e. invisible).

selection_foreground #000000

#: The foreground for text selected with the mouse. A value of none
#: means to leave the color unchanged.

selection_background #fffacd

#: The background for text selected with the mouse.


#: The 16 terminal colors. There are 8 basic colors, each color has a
#: dull and bright version. You can also set the remaining colors from
#: the 256 color table as color16 to color255.

color0 #000000
color8 #767676

#: black

color1 #cc0403
color9 #f2201f

#: red

color2  #19cb00
color10 #23fd00

#: green

color3  #cecb00
color11 #fffd00

#: yellow

color4  #0d73cc
color12 #1a8fff

#: blue

color5  #cb1ed1
color13 #fd28ff

#: magenta

color6  #0dcdcd
color14 #14ffff

#: cyan

color7  #dddddd
color15 #ffffff

#: white

#: }}}
Advanced
#: Advanced {{{

shell .

#: The shell program to execute. The default value of . means to use
#: whatever shell is set as the default shell for the current user.
#: Note that on macOS if you change this, you might need to add
#: --login to ensure that the shell starts in interactive mode and
#: reads its startup rc files.

editor .

#: The console editor to use when editing the kitty config file or
#: similar tasks. A value of . means to use the environment variable
#: EDITOR. Note that this environment variable has to be set not just
#: in your shell startup scripts but system-wide, otherwise kitty will
#: not see it.

close_on_child_death no

#: Close the window when the child process (shell) exits. If no (the
#: default), the terminal will remain open when the child exits as
#: long as there are still processes outputting to the terminal (for
#: example disowned or backgrounded processes). If yes, the window
#: will close as soon as the child process exits. Note that setting it
#: to yes means that any background processes still using the terminal
#: can fail silently because their stdout/stderr/stdin no longer work.

allow_remote_control no

#: Allow other programs to control kitty. If you turn this on other
#: programs can control all aspects of kitty, including sending text
#: to kitty windows, opening new windows, closing windows, reading the
#: content of windows, etc. Note that this even works over ssh
#: connections.

# env 

#: Specify environment variables to set in all child processes. Note
#: that environment variables are expanded recursively, so if you
#: use::

#:     env MYVAR1=a
#:     env MYVAR2=${MYVAR1}/${HOME}/b

#: The value of MYVAR2 will be a/<path to home directory>/b.

update_check_interval 0

#: Periodically check if an update to kitty is available. If an update
#: is found a system notification is displayed informing you of the
#: available update. The default is to check every 24 hrs, set to zero
#: to disable.

startup_session none

#: Path to a session file to use for all kitty instances. Can be
#: overridden by using the kitty --session command line option for
#: individual instances. See
#: https://sw.kovidgoyal.net/kitty/index.html#sessions in the kitty
#: documentation for details. Note that relative paths are interpreted
#: with respect to the kitty config directory. Environment variables
#: in the path are expanded.

clipboard_control write-clipboard write-primary

#: Allow programs running in kitty to read and write from the
#: clipboard. You can control exactly which actions are allowed. The
#: set of possible actions is: write-clipboard read-clipboard write-
#: primary read-primary. You can additionally specify no-append to
#: disable kitty's protocol extension for clipboard concatenation. The
#: default is to allow writing to the clipboard and primary selection
#: with concatenation enabled. Note that enabling the read
#: functionality is a security risk as it means that any program, even
#: one running on a remote server via SSH can read your clipboard.

term xterm-kitty

#: The value of the TERM environment variable to set. Changing this
#: can break many terminal programs, only change it if you know what
#: you are doing, not because you read some advice on Stack Overflow
#: to change it. The TERM variable is used by various programs to get
#: information about the capabilities and behavior of the terminal. If
#: you change it, depending on what programs you run, and how
#: different the terminal you are changing it to is, various things
#: from key-presses, to colors, to various advanced features may not
#: work.

#: }}}
Keyboard shortcuts
#: Keyboard shortcuts {{{

#: For a list of key names, see: GLFW keys
#: <https://www.glfw.org/docs/latest/group__keys.html>. The name to
#: use is the part after the GLFW_KEY_ prefix. For a list of modifier
#: names, see: GLFW mods
#: <https://www.glfw.org/docs/latest/group__mods.html>

#: On Linux you can also use XKB key names to bind keys that are not
#: supported by GLFW. See XKB keys
#: <https://github.com/xkbcommon/libxkbcommon/blob/master/xkbcommon/xkbcommon-
#: keysyms.h> for a list of key names. The name to use is the part
#: after the XKB_KEY_ prefix. Note that you should only use an XKB key
#: name for keys that are not present in the list of GLFW keys.

#: Finally, you can use raw system key codes to map keys. To see the
#: system key code for a key, start kitty with the kitty --debug-
#: keyboard option. Then kitty will output some debug text for every
#: key event. In that text look for ``native_code`` the value of that
#: becomes the key name in the shortcut. For example:

#: .. code-block:: none

#:     on_key_input: glfw key: 65 native_code: 0x61 action: PRESS mods: 0x0 text: 'a'

#: Here, the key name for the A key is 0x61 and you can use it with::

#:     map ctrl+0x61 something

#: to map ctrl+a to something.

#: You can use the special action no_op to unmap a keyboard shortcut
#: that is assigned in the default configuration.

#: You can combine multiple actions to be triggered by a single
#: shortcut, using the syntax below::

#:     map key combine <separator> action1 <separator> action2 <separator> action3 ...

#: For example::

#:     map kitty_mod+e combine : new_window : next_layout

#: this will create a new window and switch to the next available
#: layout

#: You can use multi-key shortcuts using the syntax shown below::

#:     map key1>key2>key3 action

#: For example::

#:     map ctrl+f>2 set_font_size 20

kitty_mod ctrl+shift

#: The value of kitty_mod is used as the modifier for all default
#: shortcuts, you can change it in your kitty.conf to change the
#: modifiers for all the default shortcuts.

clear_all_shortcuts no

#: You can have kitty remove all shortcut definition seen up to this
#: point. Useful, for instance, to remove the default shortcuts.

#: Clipboard {{{

map kitty_mod+d copy_to_clipboard

#: There is also a copy_or_interrupt action that can be optionally
#: mapped to Ctrl+c. It will copy only if there is a selection and
#: send an interrupt otherwise.

map kitty_mod+f  paste_from_clipboard
map kitty_mod+F  paste_from_selection
map shift+insert paste_from_selection
map kitty_mod+o  pass_selection_to_program

#: You can also pass the contents of the current selection to any
#: program using pass_selection_to_program. By default, the system's
#: open program is used, but you can specify your own, the selection
#: will be passed as a command line argument to the program, for
#: example::

#:     map kitty_mod+o pass_selection_to_program firefox

#: You can pass the current selection to a terminal program running in
#: a new kitty window, by using the @selection placeholder::

#:     map kitty_mod+y new_window less @selection

#: }}}

#: Scrolling {{{

map kitty_mod+up        scroll_line_up
map kitty_mod+k         scroll_line_up
map kitty_mod+down      scroll_line_down
map kitty_mod+j         scroll_line_down
map kitty_mod+page_up   scroll_page_up
map kitty_mod+page_down scroll_page_down

map kitty_mod+l   scroll_page_up
map kitty_mod+; scroll_page_down

map kitty_mod+home      scroll_home
map kitty_mod+end       scroll_end
map kitty_mod+h         show_scrollback

#: You can pipe the contents of the current screen + history buffer as
#: STDIN to an arbitrary program using the ``pipe`` function. For
#: example, the following opens the scrollback buffer in less in an
#: overlay window::

#:     map f1 pipe @ansi overlay less +G -R

#: For more details on piping screen and buffer contents to external
#: programs, see pipe.

#: }}}

#: Window management {{{

map kitty_mod+enter new_window

#: You can open a new window running an arbitrary program, for
#: example::

#:     map kitty_mod+y      new_window mutt

#: You can open a new window with the current working directory set to
#: the working directory of the current window using::

#:     map ctrl+alt+enter    new_window_with_cwd

#: You can open a new window that is allowed to control kitty via the
#: kitty remote control facility by prefixing the command line with @.
#: Any programs running in that window will be allowed to control
#: kitty. For example::

#:     map ctrl+enter new_window @ some_program

#: You can open a new window next to the currently active window or as
#: the first window, with::

#:     map ctrl+n new_window !neighbor some_program
#:     map ctrl+f new_window !first some_program

map kitty_mod+n new_os_window
map kitty_mod+w close_window

#map kitty_mod+] next_window
#map kitty_mod+[ previous_window
map kitty_mod+j next_window
map kitty_mod+k previous_window

map kitty_mod+f move_window_forward
map kitty_mod+b move_window_backward
map kitty_mod+` move_window_to_top
map kitty_mod+r start_resizing_window
map kitty_mod+1 first_window
map kitty_mod+2 second_window
map kitty_mod+3 third_window
map kitty_mod+4 fourth_window
map kitty_mod+5 fifth_window
map kitty_mod+6 sixth_window
map kitty_mod+7 seventh_window
map kitty_mod+8 eighth_window
map kitty_mod+9 ninth_window
map kitty_mod+0 tenth_window
#: }}}

#: Tab management {{{

map kitty_mod+right next_tab
map kitty_mod+o next_tab
map kitty_mod+left  previous_tab
map kitty_mod+i  previous_tab
map kitty_mod+t     new_tab
map kitty_mod+q     close_tab
map kitty_mod+.     move_tab_forward
map kitty_mod+,     move_tab_backward
map kitty_mod+alt+t set_tab_title

#: You can also create shortcuts to go to specific tabs, with 1 being
#: the first tab, 2 the second tab and -1 being the previously active
#: tab::

#:     map ctrl+alt+1 goto_tab 1
#:     map ctrl+alt+2 goto_tab 2

#: Just as with new_window above, you can also pass the name of
#: arbitrary commands to run when using new_tab and use
#: new_tab_with_cwd. Finally, if you want the new tab to open next to
#: the current tab rather than at the end of the tabs list, use::

#:     map ctrl+t new_tab !neighbor [optional cmd to run]
#: }}}

#: Layout management {{{

map kitty_mod+space next_layout

#: You can also create shortcuts to switch to specific layouts::

#:     map ctrl+alt+t goto_layout tall
#:     map ctrl+alt+s goto_layout stack

#: Similarly, to switch back to the previous layout::

#:    map ctrl+alt+p last_used_layout
#: }}}

#: Font sizes {{{

#: You can change the font size for all top-level kitty OS windows at
#: a time or only the current one.

map kitty_mod+equal     change_font_size all +2.0
map kitty_mod+minus     change_font_size all -2.0
map kitty_mod+backspace change_font_size all 0

#: To setup shortcuts for specific font sizes::

#:     map kitty_mod+f6 change_font_size all 10.0

#: To setup shortcuts to change only the current OS window's font
#: size::

#:     map kitty_mod+f6 change_font_size current 10.0
#: }}}

#: Select and act on visible text {{{

#: Use the hints kitten to select text and either pass it to an
#: external program or insert it into the terminal or copy it to the
#: clipboard.

#map kitty_mod+e kitten hints

#: Open a currently visible URL using the keyboard. The program used
#: to open the URL is specified in open_url_with.

map kitty_mod+h>f kitten hints --type path --program default

#: Select a path/filename and insert it into the terminal. Useful, for
#: instance to run git commands on a filename output from a previous
#: git command.

#map kitty_mod+p>shift+f kitten hints --type path
map kitty_mod+h>p kitten hints --type path --program default

#: Select a path/filename and open it with the default open program.

#map kitty_mod+p>l kitten hints --type line --program -
map ctrl+shift+h>l kitten hints --type line --program *

#: Select a line of text and insert it into the terminal. Use for the
#: output of things like: ls -1

#map kitty_mod+p>w kitten hints --type word --program -
map ctrl+shift+h>w kitten hints --type word --program *

#: Select words and insert into terminal.

#: Select something that looks like a hash and insert it into the
#: terminal. Useful with git, which uses sha1 hashes to identify
#: commits


#: The hints kitten has many more modes of operation that you can map
#: to different shortcuts. For a full description see kittens/hints.
#: }}}

#: Miscellaneous {{{

map kitty_mod+f11    toggle_fullscreen
map kitty_mod+f10    toggle_maximized
map kitty_mod+u      kitten unicode_input
map kitty_mod+f2     edit_config_file
map kitty_mod+escape kitty_shell window

#: Open the kitty shell in a new window/tab/overlay/os_window to
#: control kitty using commands.

map kitty_mod+a>m    set_background_opacity +0.1
map kitty_mod+a>l    set_background_opacity -0.1
map kitty_mod+a>1    set_background_opacity 1
map kitty_mod+a>d    set_background_opacity default
map kitty_mod+delete clear_terminal reset active

#: You can create shortcuts to clear/reset the terminal. For example::

#:     # Reset the terminal
#:     map kitty_mod+f9 clear_terminal reset active
#:     # Clear the terminal screen by erasing all contents
#:     map kitty_mod+f10 clear_terminal clear active
#:     # Clear the terminal scrollback by erasing it
#:     map kitty_mod+f11 clear_terminal scrollback active
#:     # Scroll the contents of the screen into the scrollback
#:     map kitty_mod+f12 clear_terminal scroll active

#: If you want to operate on all windows instead of just the current
#: one, use all instead of active.

#: It is also possible to remap Ctrl+L to both scroll the current
#: screen contents into the scrollback buffer and clear the screen,
#: instead of just clearing the screen::

#:     map ctrl+l combine : clear_terminal scroll active : send_text normal,application \x0c


#: You can tell kitty to send arbitrary (UTF-8) encoded text to the
#: client program when pressing specified shortcut keys. For example::

#:     map ctrl+alt+a send_text all Special text

#: This will send "Special text" when you press the ctrl+alt+a key
#: combination.  The text to be sent is a python string literal so you
#: can use escapes like \x1b to send control codes or \u21fb to send
#: unicode characters (or you can just input the unicode characters
#: directly as UTF-8 text). The first argument to send_text is the
#: keyboard modes in which to activate the shortcut. The possible
#: values are normal or application or kitty or a comma separated
#: combination of them.  The special keyword all means all modes. The
#: modes normal and application refer to the DECCKM cursor key mode
#: for terminals, and kitty refers to the special kitty extended
#: keyboard protocol.

#: Another example, that outputs a word and then moves the cursor to
#: the start of the line (same as pressing the Home key)::

#:     map ctrl+alt+a send_text normal Word\x1b[H
#:     map ctrl+alt+a send_text application Word\x1bOH

#: }}}

# }}}

Mouse

Toggle trackball

Uses hhpc utility ../Dist/hhpc-master/

[[ -f /tmp/trackball-disabled ]] && STATE="disabled" || STATE="enabled"

if [[ "$1" == "kill" ]]; then
    killall hhpc > /dev/null 2>&1;
    rm -f /tmp/trackball-disabled
    echo "mouse.coords({x=960,y=540})" | awesome-client
    exit
fi

if [[ "$1" == "status" ]]; then
    echo $STATE
    exit
fi

[[ "$STATE" == "disabled" ]] && NEW_STATE="enabled" || NEW_STATE="disabled"
[[ "$1" == "enable" ]] && NEW_STATE="enabled"
[[ "$1" == "disable" ]] && NEW_STATE="disabled"

if [[ ! -z "`pgrep hhpc`" ]]; then
    [[ "$NEW_STATE" == "$STATE" ]] && exit
fi

if [[ "$NEW_STATE" == "enabled" ]]; then
    killall hhpc 2>&1 > /dev/null;
    rm -f /tmp/trackball-disabled
    echo "mouse.coords({x=960,y=540})" | awesome-client
else
    hhpc -i 0&
    touch /tmp/trackball-disabled
fi

Clipboard ring integrated with Rofi quick menu: clipmenu

x11-misc/clipmenu
x11-misc/clipmenu ~amd64
=x11-misc/clipnotify-1.0.2 ~amd64

Run daemon on startup:

DISPLAY=:0 clipmenud >/dev/null 2>&1 &

Script to show clipboard history and paste it at point. Uses xdotool (to actually insert) and clipmenu

CM_LAUNCHER=rofi clipmenu -width -150 && xdotool key Shift+Insert

The script is bound in Awesome WM settings.

App wrappers

Collection of bash script invoking default apps (with possible tweaks)

Music player

Show playlist:

emacsclient -F "'(fullscreen . maximized)" -n -c -e "(mingus)"

Delete currently playing song from the playlist:

emacsclient -e "(yant/mingus-delete-currently-playing)"

State popup

#disabled
exit;

ps aux | grep "[n]cmpcpp --current-song %a «%b»" && kill `ps aux | grep "[n]cmpcpp --current-song %a «%b»" | cut -f2 -d' '` 2>&1 > /dev/null
[[ -e /tmp/mpd_state ]] || echo "`ncmpcpp --current-song "%a «%b»"`" "`ncmpcpp --current-song "(%l) - %t"`" > /tmp/mpd_state
echo "`ncmpcpp --current-song "%a «%b»"`" "`ncmpcpp --current-song "(%l) - %t"`" > /tmp/mpd_state_new
diff /tmp/mpd_state /tmp/mpd_state_new 2>&1 > /dev/null|| if [ true ]; then
	[[ -z "`ncmpcpp --current-song | grep "mpd"`" ]] && notify-send "`ncmpcpp --current-song "%a «%b»"`" "`ncmpcpp --current-song "(%l) - %t"`"
	echo "`ncmpcpp --current-song "%a «%b»"`" "`ncmpcpp --current-song "(%l) - %t"`" > /tmp/mpd_state
fi

Pause/play

mpc -q toggle && mpd_state_popup.sh

Next

mpc -q next

Previous

mpc -q prev

Next

mpc -q next

Mute

amixer -D pulse -q set Master toggle
volstatus_pop.sh

Increase volume

amixer -q set Master 5%+

VOLUME="`amixer sget Master | tail -n1 | cut -d[ -f 2 | cut -d] -f 1 | cut -d% -f1 | head -n 1`"
VOLUME_STATUS="`amixer sget Master | tail -n1 | cut -d[ -f 4 | cut -d] -f1 | head -n 1`"

pacmd list-sinks | grep BATBAND && pacmd set-sink-volume $(pacmd list-sinks | grep index | tail -n1 | cut -d: -f2) $(echo "$VOLUME * 855" | bc -l) # 65536 max * 0.01

Decrease volume

amixer -q set Master 5%-

VOLUME="`amixer sget Master | tail -n1 | cut -d[ -f 2 | cut -d] -f 1 | cut -d% -f1 | head -n 1`"
VOLUME_STATUS="`amixer sget Master | tail -n1 | cut -d[ -f 4 | cut -d] -f1 | head -n 1`"

pacmd list-sinks | grep BATBAND && pacmd set-sink-volume $(pacmd list-sinks | grep index | tail -n1 | cut -d: -f2) $(echo "$VOLUME * 855" | bc -l) # 65536 max * 0.01

Email client

searchstring="tag:inbox and tag:todo"
[[ "$1" == "--all" ]] && searchstring="tag:todo"

[[ "$1" == "--newsonly" ]] && searchstring="tag:listinbox and tag:todo"
[[ "$1" == "--inbox" || "$1" == "--newsonly" ]] && emacsclient -n -c -e "(progn (notmuch-search \"$searchstring\") (setq notmuch-frame t))" || emacsclient -n -c -e "(progn (notmuch-mua-new-mail) (setq notmuch-frame t))"

RSS reader

# emacsclient -F "'(fullscreen . maximized)" -c -e "(require 'elfeed-org)" -e "(elfeed-org)" -e "(elfeed)" -e "(sleep-for 0 10)" -e "(elfeed-search-update--force)"
emacsclient -F "'(fullscreen . maximized)" -c -e "(require 'elfeed-org)" -e "(elfeed-org)" -e "(elfeed)"

Text editor

dev-libs/jemalloc
# [[ "$#" == "0" ]] && emacsclient --frame-parameters='(quote (name . "main"))' -c -n -e '(switch-to-buffer (quote "*scratch*"))' && exit
[[ "$#" == "0" ]] && emacsclient -c -n -e '(switch-to-buffer (quote "*scratch*"))' -e '(boon-set-command-state)' >>/dev/null 2>&1  || \
	#    /home/yantar92/tmp/emacs2/emacs/src/emacs && exit
	# emacs && exit
	~/Git/emacs/src/emacs && exit
[[ "$1" == "agenda" ]] && emacsclient -c -n -e '(org-agenda nil " ")'
#run_single_window.sh "Emacs" "sh -c emacs -name \"main\""

Browser

qutebrowser --loglevel critical --backend webengine 2>&1 | qutebrowser-logger.sh
while read x; do
    [[ "$x" == *"with Emacs Client"* ]] && continue
    [[ "$x" == *"Path override failed for key base::DIR_APP_DICTIONARIES"* ]] && continue
    [[ "$x" == *"Check failed: false"* ]] && continue
    [[ "$x" == "Waiting for Emacs..." ]] && continue
    [[ "$x" == *"libpng warning:"* ]] && continue
    [[ "$x" == *"Fontconfig error: Cannot load default config file: No such file: (null)"* ]] && continue
    [[ "$x" == *"handshake failed; returned -1, SSL error code 1"* ]] && continue
    [[ "$x" == *"Message 4 rejected by interface"* ]] && continue
    [[ "$x" == *"Message 5 rejected by interface"* ]] && continue
    [[ "$x" == *"org.freedesktop.DBus.Error.ServiceUnknown: The name org.freedesktop.UPower was not provided by any .service files"* ]] && continue
    [[ "$x" == "" ]] && continue
    notify-send "Qutebrowser: $x"
    echo $(date): $x >> ~/.log/qutebrowser.log
done

Mime info tools

xdg-utils

x11-misc/xdg-utils

Shell: bash

Configuration

Common config to be used in interactive and noninteractive shells

export PATH="/home/yantar92/.local/bin:$PATH"
export DE="generic"
export GNUPLOT_LIB="~/.gnuplot/layout:$GNUPLOT_LIB"
export EDITOR="emacsclient -c"
export VISUAL="$EDITOR"

export PASSWORD_STORE_ENABLE_EXTENSIONS="true"

alias cleanlinks="true" # do not want to use it

Bash config

source ~/.profile

# Infinite history
export HISTSIZE=-1
export HISTFILESIZE=-1
export HISTIGNORE="&:ls:[bf]g:exit"
export PROMPT_COMMAND='history -a'

shopt -s cdspell
shopt -s histappend
set bell-style none

alias ls="ls --color"
alias la="ls -a"
alias ll="ls -lh"
alias lal="la -alh"
alias sl="ls"
alias cl="cls"
alias csl="cls"
alias bc="bc -l"
alias rsync="rsync -P"

if [ -f /usr/bin/grc ]; then
    alias ping="grc ping"
    alias gcc="grc gcc"
    alias g++="grc g++"
    alias make="grc make"
    # alias cat="grc cat"
    # alias head="grc head"
    alias tail="grc tail"
    alias diff="grc diff"
fi

complete -cf sudo
complete -cf man

#trying to fix "Inappropriate ioctl for device" error from gpg
#https://github.com/keybase/keybase-issues/issues/1712
export GPG_TTY=$(tty)

#prompt
source ~/Dist/liquidprompt/liquidprompt
LP_PS1_POSTFIX="\n> "

# Emacs eat-terminal support
[ -n "$EAT_SHELL_INTEGRATION_DIR" ] && \
  source "$EAT_SHELL_INTEGRATION_DIR/bash"

../Dist/liquidprompt/

make ebuild for liquidpromptEND

Bash completion (system-wide)

Need to install bash-completion

app-shells/bash-completion

Qutebrowser

Files to copy:

  • file:~/.config/qutebrowser/bookmarks/urls
  • file:~/.local/share/qutebrowser/cmd-history
  • file:~/.local/share/qutebrowser/history.sqlite

[2024-04-26 Fri] qtwebengine 6.7.0 fails to compile due to bug on Gentoo https://bugs.gentoo.org/930107. Masking temporarily.

# =dev-qt/qtwebengine-6.7.0
=dev-build/ninja-1.12.0
www-client/qutebrowser

Use cutting edge version.

www-client/qutebrowser **

Use built-in scripts.

www-client/qutebrowser scripts

Userscripts

Open video link using mpv + youtube-dl

rawurlencode() {
  local string="${1}"
  local strlen=${#string}
  local encoded=""
  local pos c o

  for (( pos=0 ; pos<strlen ; pos++ )); do
     c=${string:$pos:1}
     case "$c" in
         [-_.~a-zA-Zа-яА-я0-9] ) o="${c}" ;;
	 [\[\]] ) o="|" ;;
         * )               printf -v o '%%%02x' "'$c"
     esac
     encoded+="${o}"
  done
#  echo "${encoded}"    # You can either set a return variable (FASTER) 
  REPLY="${encoded}"   #+or echo the result (EASIER)... or both... :p
}

rawurlencode "$1"
LINK="$REPLY"

echo "message-info \"Adding $QUTE_URL to mpv... \"" >> "$QUTE_FIFO" && mpv "$1" && echo "message-info \"Adding to $QUTE_URL mpv... done\"" >> "$QUTE_FIFO" || echo "message-error \"Adding $QUTE_URL to mpv... failed\"" >> "$QUTE_FIFO"

Org-mode capture

rawurlencode() {
  local string="${1}"
  local strlen=${#string}
  local encoded=""
  local pos c o

  for (( pos=0 ; pos<strlen ; pos++ )); do
     c=${string:$pos:1}
     case "$c" in
         [-_.~a-zA-Zа-яА-Я0-9] ) o="${c}" ;;
	 [\[\]] ) o="|" ;;
	 * )               printf -v o '%%%02x' "'$c"
     esac
     encoded+="${o}"
  done
  echo "${encoded}"    # You can either set a return variable (FASTER) 
  REPLY="${encoded}"   #+or echo the result (EASIER)... or both... :p
}

# Returns a string in which the sequences with percent (%) signs followed by
# two hex digits have been replaced with literal characters.
rawurldecode() {

  # This is perhaps a risky gambit, but since all escape characters must be
  # encoded, we can replace %NN with \xNN and pass the lot to printf -b, which
  # will decode hex for us

  printf -v REPLY '%b' "${1//%/\\x}" # You can either set a return variable (FASTER)

#  echo "${REPLY}"  #+or echo the result (EASIER)... or both... :p
}


# Initialize all the option variables.
# This ensures we are not contaminated by variables from the environment.
TEMPLATE="b"
FORCE=""

while :; do
    case $1 in
	--force)       # Takes an option argument; ensure it has been specified.
	    FORCE="t"
	    shift
            ;;
        --silent)
	    TEMPLATE="B"
            shift
            ;;
        --rss)
            TEMPLATE="r"
            shift
            ;;
        *)
            break
    esac
    shift 
done 
 
rawurlencode "$QUTE_URL"
# QUTE_URL=$(echo $QUTE_URL | sed -r 's/^[^/]+//')
URL="$REPLY"
# URL="$QUTE_URL"
# rawurlencode "$QUTE_TITLE"
# TITLE="$REPLY"
TITLE="$(echo $QUTE_TITLE | sed -r 's/&//g')"
# rawurlencode "$QUTE_SELECTED_TEXT"
# # SELECTED_TEXT="$REPLY"
SELECTED_TEXT="$QUTE_SELECTED_TEXT"
#emacsclient "org-protocol://capture://b/$URL/$TITLE/$SELECTED_TEXT" && echo "jseval \"Bookmark saved to TODO.org/Inbox\"" >> "$QUTE_FIFO" || echo "jseval \"Bookmark not saved!\"" >> "$QUTE_FIFO"

# CHECK="$(org-check-bookmark.sh "$QUTE_URL")"

# echo "message-warning \"org-protocol://capture?template=b&url=$URL&title=$TITLE&body=$SELECTED_TEXT"\"  >> "$QUTE_FIFO"
# if [[ "$CHECK" == "ok" || "$FORCE" == "t" ]]; then 
# (emacsclient "org-protocol://capture?template=$TEMPLATE&url=$URL&title=$TITLE&body=$SELECTED_TEXT&html=$QUTE_HTML&qutebrowser-fifo=$QUTE_FIFO"\
    #      && echo "message-info '$(cat ~/Org/inbox.org | grep \* | tail -n1)'" >> "$QUTE_FIFO" || echo "message-error \"Bookmark not saved!\"" >> "$QUTE_FIFO");
emacsclient "org-protocol://capture?template=$TEMPLATE&url=$URL&title=$TITLE&body=$SELECTED_TEXT&html=$QUTE_HTML&qutebrowser-fifo=$QUTE_FIFO" && true;
# else
#     echo -e "message-warning \"$CHECK\"" >> "$QUTE_FIFO"
# fi
# grep "$QUTE_URL" ~/Org/TODO.org* ~/Org/inbox.org ~/Org/notes.org* >/dev/null && (echo "message-warning \"Url is already in TODO.org*\"" >> "$QUTE_FIFO") ||\
    #     (emacsclient "org-protocol://capture?template=b&url=$URL&title=$TITLE&body=$SELECTED_TEXT"\
    # 	 && echo "message-info \"Bookmark saved to inbox.org/Inbox\"" >> "$QUTE_FIFO" || echo "message-error \"Bookmark not saved!\"" >> "$QUTE_FIFO")
org-protocol-add-bookmark.sh --force
rawurlencode() {
    local string="${1}"
    local strlen=${#string}
    local encoded=""
    local pos c o

    for (( pos=0 ; pos<strlen ; pos++ )); do
	c=${string:$pos:1}
	case "$c" in
            [-_.~a-zA-Zа-яА-я0-9] ) o="${c}" ;;
	    [\[\]] ) o="|" ;;
            * )               printf -v o '%%%02x' "'$c"
	esac
	encoded+="${o}"
    done
    echo "${encoded}"    # You can either set a return variable (FASTER) 
    REPLY="${encoded}"   #+or echo the result (EASIER)... or both... :p
}

# Returns a string in which the sequences with percent (%) signs followed by
# two hex digits have been replaced with literal characters.
rawurldecode() {

    # This is perhaps a risky gambit, but since all escape characters must be
    # encoded, we can replace %NN with \xNN and pass the lot to printf -b, which
    # will decode hex for us

    printf -v REPLY '%b' "${1//%/\\x}" # You can either set a return variable (FASTER)

    #  echo "${REPLY}"  #+or echo the result (EASIER)... or both... :p
}

rawurlencode "$QUTE_URL"
URL="$REPLY"
rawurlencode "$QUTE_TITLE"
TITLE="$REPLY"
rawurlencode "$QUTE_SELECTED_TEXT"
SELECTED_TEXT="$REPLY"
#emacsclient "org-protocol://capture://b/$URL/$TITLE/$SELECTED_TEXT" && echo "jseval \"Bookmark saved to TODO.org/Inbox\"" >> "$QUTE_FIFO" || echo "jseval \"Bookmark not saved!\"" >> "$QUTE_FIFO"
emacsclient "org-protocol://capture?template=s&url=$URL&title=$TITLE&body=$SELECTED_TEXT" && echo "message-info \"Selection saved to TODO.org/Inbox\"" >> "$QUTE_FIFO" || echo "message-error \"Selection not saved!\"" >> "$QUTE_FIFO"

JS (greasemonkey) scripts

Hide ad popup in nytimes.com

function hidepopup() {
    if (window.location.host == 'www.nytimes.com' ) {
	var adPopup = document.getElementsByClassName('MAG_web_all_Monthly-Sale-dock')[0];
	if (adPopup) {
	    adPopup.parentNode.removeChild(adPopup);
	}}
};

window.addEventListener(
    'load',
    function() {window.setTimeout(hidepopup, 800)},
    true)

Quickmarks

Custom high-priority bookmarks

RSSHub http://127.0.0.1:1200/

Configuration

Load interactively set preferences

# Uncomment this to still load settings configured via autoconfig.yml
config.load_autoconfig()

Ad-blocker

# Block Youtube advertisement
# Credit: https://gist.github.com/Gavinok/f9c310a66576dc00329dd7bef2b122a1
from qutebrowser.api import interceptor

"""
qutebrowser settings for video
for more settings check out
https://qutebrowser.org/doc/help/settings.html
"""

# ================== Youtube Add Blocking ======================= {{{
def filter_yt(info: interceptor.Request):
    """Block the given request if necessary."""
    url = info.request_url
    if (
        url.host() == "www.youtube.com"
        and url.path() == "/get_video_info"
        and "&adformat=" in url.query()
    ):
        info.block()


interceptor.register(filter_yt)

Do not open new links in separate frames

c.new_instance_open_target="tab"
c.tabs.tabs_are_windows=False

Default page

c.url.default_page = 'http://127.0.0.1:8384/'
c.url.start_pages = 'http://127.0.0.1:8384/'

Session

# Always restore open sites when qutebrowser is reopened.
# Type: Bool
c.auto_save.session = False

Security

# Automatically start playing `<video>` elements. Note: On Qt < 5.11,
# this option needs a restart and does not support URL patterns.
# Type: Bool
c.content.autoplay = False

# Allow websites to read canvas elements. Note this is needed for some
# websites to work properly.
# Type: Bool
c.content.canvas_reading = True
# True is required for youtube player to work

c.content.headers.accept_language="en-US,en;q=0.5"
c.content.headers.custom={"accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"}

# Which cookies to accept. With QtWebEngine, this setting also controls
# other features with tracking capabilities similar to those of cookies;
# including IndexedDB, DOM storage, filesystem API, service workers, and
# AppCache. Note that with QtWebKit, only `all` and `never` are
# supported as per-domain values. Setting `no-3rdparty` or `no-
# unknown-3rdparty` per-domain on QtWebKit will have the same effect as
# `all`.
# Type: String
# Valid values:
#   - all: Accept all cookies.
#   - no-3rdparty: Accept cookies from the same origin only. This is known to break some sites, such as GMail.
#   - no-unknown-3rdparty: Accept cookies from the same origin only, unless a cookie is already set for the domain. On QtWebEngine, this is the same as no-3rdparty.
#   - never: Don't accept cookies at all.
c.content.cookies.accept = 'all'

# Which cookies to accept. With QtWebEngine, this setting also controls
# other features with tracking capabilities similar to those of cookies;
# including IndexedDB, DOM storage, filesystem API, service workers, and
# AppCache. Note that with QtWebKit, only `all` and `never` are
# supported as per-domain values. Setting `no-3rdparty` or `no-
# unknown-3rdparty` per-domain on QtWebKit will have the same effect as
# `all`.
# Type: String
# Valid values:
#   - all: Accept all cookies.
#   - no-3rdparty: Accept cookies from the same origin only. This is known to break some sites, such as GMail.
#   - no-unknown-3rdparty: Accept cookies from the same origin only, unless a cookie is already set for the domain. On QtWebEngine, this is the same as no-3rdparty.
#   - never: Don't accept cookies at all.
config.set('content.cookies.accept', 'all', 'chrome-devtools://*')

config.set('content.cookies.accept', 'all', 'https://127.0.0.1/*')
config.set('content.cookies.accept', 'all', 'https://*.ylab.cn/*')
config.set('content.cookies.accept', 'all', 'https://*.wechat.com/*')
config.set('content.cookies.accept', 'all', 'https://*.microsoftonline.com/*')
config.set('content.cookies.accept', 'all', 'https://*.office.com/*')
config.set('content.cookies.accept', 'all', 'https://*.amazon.cn/*')
config.set('content.cookies.accept', 'all', 'https://*.amazon.com/*')
config.set('content.cookies.accept', 'all', 'https://*.rutracker.org/*')
config.set('content.cookies.accept', 'all', 'https://*.chess.com/*')
config.set('content.cookies.accept', 'all', 'https://*.facebook.com/*')
config.set('content.cookies.accept', 'all', 'https://*.reddit.com/*')
config.set('content.cookies.accept', 'all', 'https://*.author.today/*')
config.set('content.cookies.accept', 'all', 'https://*.ficbook.net/*')
config.set('content.cookies.accept', 'all', 'https://*.github.com/*')
config.set('content.cookies.accept', 'all', 'https://*.gitlab.com/*')
config.set('content.cookies.accept', 'all', 'https://*.baidu.com/*')
config.set('content.cookies.accept', 'all', 'https://*.weibo.com/*')
config.set('content.cookies.accept', 'all', 'https://*.bilibili.com/*')
config.set('content.cookies.accept', 'all', 'https://*.google.com/*')
config.set('content.cookies.accept', 'all', 'https://*.google.com.sg/*')
config.set('content.cookies.accept', 'all', 'https://*.youtube.com/*')
config.set('content.cookies.accept', 'all', 'https://youtube.com/*')
config.set('content.cookies.accept', 'all', 'https://scholar.google.com/*')
config.set('content.cookies.accept', 'all', 'https://*.chrono.gg/*')
config.set('content.cookies.accept', 'all', 'https://*.paypal.com/*')
config.set('content.cookies.accept', 'all', 'https://*.acs.org/*')
config.set('content.cookies.accept', 'all', 'https://*.scitation.org/*')
config.set('content.cookies.accept', 'all', 'https://*.tandfonline.com/*')
config.set('content.cookies.accept', 'all', 'https://*.wiley.com/*')
config.set('content.cookies.accept', 'all', 'https://*.owncube.com/*')
config.set('content.cookies.accept', 'all', 'https://*.linkedin.com/*')

# Which cookies to accept. With QtWebEngine, this setting also controls
# other features with tracking capabilities similar to those of cookies;
# including IndexedDB, DOM storage, filesystem API, service workers, and
# AppCache. Note that with QtWebKit, only `all` and `never` are
# supported as per-domain values. Setting `no-3rdparty` or `no-
# unknown-3rdparty` per-domain on QtWebKit will have the same effect as
# `all`.
# Type: String
# Valid values:
#   - all: Accept all cookies.
#   - no-3rdparty: Accept cookies from the same origin only. This is known to break some sites, such as GMail.
#   - no-unknown-3rdparty: Accept cookies from the same origin only, unless a cookie is already set for the domain. On QtWebEngine, this is the same as no-3rdparty.
#   - never: Don't accept cookies at all.
config.set('content.cookies.accept', 'all', 'devtools://*')

# Store cookies. Note this option needs a restart with QtWebEngine on Qt
# < 5.9.
# Type: Bool
c.content.cookies.store = True

# Which cookies to accept. With QtWebEngine, this setting also controls
# other features with tracking capabilities similar to those of cookies;
# including IndexedDB, DOM storage, filesystem API, service workers, and
# AppCache. Note that with QtWebKit, only `all` and `never` are
# supported as per-domain values. Setting `no-3rdparty` or `no-
# unknown-3rdparty` per-domain on QtWebKit will have the same effect as
# `all`.
# Type: String
# Valid values:
#   - all: Accept all cookies.
#   - no-3rdparty: Accept cookies from the same origin only. This is known to break some sites, such as GMail.
#   - no-unknown-3rdparty: Accept cookies from the same origin only, unless a cookie is already set for the domain. On QtWebEngine, this is the same as no-3rdparty.
#   - never: Don't accept cookies at all.
config.set('content.cookies.accept', 'all', 'chrome-devtools://*')

# Which cookies to accept. With QtWebEngine, this setting also controls
# other features with tracking capabilities similar to those of cookies;
# including IndexedDB, DOM storage, filesystem API, service workers, and
# AppCache. Note that with QtWebKit, only `all` and `never` are
# supported as per-domain values. Setting `no-3rdparty` or `no-
# unknown-3rdparty` per-domain on QtWebKit will have the same effect as
# `all`.
# Type: String
# Valid values:
#   - all: Accept all cookies.
#   - no-3rdparty: Accept cookies from the same origin only. This is known to break some sites, such as GMail.
#   - no-unknown-3rdparty: Accept cookies from the same origin only, unless a cookie is already set for the domain. On QtWebEngine, this is the same as no-3rdparty.
#   - never: Don't accept cookies at all.
config.set('content.cookies.accept', 'all', 'devtools://*')

# Try to pre-fetch DNS entries to speed up browsing.
# Type: Bool
c.content.dns_prefetch = True



# Load images automatically in web pages.
# Type: Bool
config.set('content.images', True, 'chrome-devtools://*')

# Load images automatically in web pages.
# Type: Bool
config.set('content.images', True, 'devtools://*')

# Enable JavaScript.
# Type: Bool
c.content.javascript.enabled = True

config.set('content.javascript.enabled', True, 'https://*.amazon.cn/*')
config.set('content.javascript.enabled', True, 'https://*.amazon.com/*')
config.set('content.javascript.enabled', True, 'https://*.rutracker.org/*')
config.set('content.javascript.enabled', True, 'https://*.facebook.com/*')
config.set('content.javascript.enabled', True, 'https://fanyi.baidu.com/*')
config.set('content.javascript.enabled', True, 'https://*.bilibili.com/*')
config.set('content.javascript.enabled', True, 'https://www.fedex.com/*')
config.set('content.javascript.enabled', True, 'http://campnano.xjtu.ylab.cn/*')
config.set('content.javascript.enabled', True, 'https://www.humblebundle.com/*')
config.set('content.javascript.enabled', True, 'https://torguard.net/*')
config.set('content.javascript.enabled', True, 'https://mail.google.com/*')
config.set('content.javascript.enabled', True, 'https://docs.google.com/*')
config.set('content.javascript.enabled', True, 'https://scholar.google.com.sg/*')
config.set('content.javascript.enabled', True, 'https://duckduckgo.com/*')
config.set('content.javascript.enabled', True, 'https://author.today/*')
config.set('content.javascript.enabled', True, 'https://www.chess.com/*')
config.set('content.javascript.enabled', True, 'https://www.sciencedirect.com/*')
config.set('content.javascript.enabled', True, 'https://reader.elsevier.com/*')
config.set('content.javascript.enabled', True, 'https://*.ficbook.net/*')
config.set('content.javascript.enabled', True, 'https://www.dropbox.com/*')
config.set('content.javascript.enabled', True, 'https://www.researchgate.net/*')
config.set('content.javascript.enabled', True, 'https://litnet.com/*')
config.set('content.javascript.enabled', True, 'https://*.reddit.com/*')
config.set('content.javascript.enabled', True, 'https://mail.humblebundle.com/*')
config.set('content.javascript.enabled', True, 'https://*.dbs.com.sg/*')
config.set('content.javascript.enabled', True, 'https://www.paypal.com/*')
config.set('content.javascript.enabled', True, 'https://*.steampowered.com/*')
config.set('content.javascript.enabled', True, 'https://www.alibaba.com/*')
config.set('content.javascript.enabled', True, 'https://*.office.com/*')
config.set('content.javascript.enabled', True, 'https://*.microsoftonline.com/*')
config.set('content.javascript.enabled', True, 'https://*.mfa.gov.ua/*')
config.set('content.javascript.enabled', True, 'http://www.china-embassy.org/*')
config.set('content.javascript.enabled', True, 'https://*.sutd.edu.sg/*')
config.set('content.javascript.enabled', True, 'https://sutd.edu.sg/*')
config.set('content.javascript.enabled', True, 'https://*.grammarly.com/*')
config.set('content.javascript.enabled', True, 'https://calendly.com/*')
config.set('content.javascript.enabled', True, 'https://web.whatsapp.com/*')
config.set('content.javascript.enabled', True, 'https://www.linuxcounter.net/*')
config.set('content.javascript.enabled', True, 'http://www.editorialmanager.com/*')
config.set('content.javascript.enabled', True, 'https://www.editorialmanager.com/*')
config.set('content.javascript.enabled', True, 'https://orcid.org/*')
config.set('content.javascript.enabled', True, 'https://webshop.elsevier.com/*')
config.set('content.javascript.enabled', True, 'https://*.ica.gov.sg/*')
config.set('content.javascript.enabled', True, 'https://*.esplanade.com/*')
config.set('content.javascript.enabled', True, 'https://www.tripadvisor.com.sg/*')
config.set('content.javascript.enabled', True, 'https://www.last.fm/*')
config.set('content.javascript.enabled', True, 'https://www.twitch.tv/*')
config.set('content.javascript.enabled', True, 'https://emploi.epfl.ch/*')
config.set('content.javascript.enabled', True, 'https://www.sharelatex.com/*')
config.set('content.javascript.enabled', True, 'https://v2.overleaf.com/*')
config.set('content.javascript.enabled', True, 'https://*.express.dhl')
config.set('content.javascript.enabled', True, 'https://gitlab.com')
config.set('content.javascript.enabled', True, 'https://*.edx.org/*')
config.set('content.javascript.enabled', True, 'https://courses.edx.org/*')
config.set('content.javascript.enabled', True, 'https://edx.org')
config.set('content.javascript.enabled', True, 'https://*.patreon.com')
config.set('content.javascript.enabled', True, 'https://*.aps.org')
config.set('content.javascript.enabled', True, 'https://*.acs.org')
config.set('content.javascript.enabled', True, 'https://www.lesswrong.com')
config.set('content.javascript.enabled', True, 'https://www.skyscanner.com.sg/*')
config.set('content.javascript.enabled', True, 'https://*.booking.com')
config.set('content.javascript.enabled', True, 'https://*.fun-mooc.fr/*')
config.set('content.javascript.enabled', True, 'https://web.wechat.com/*')
config.set('content.javascript.enabled', True, 'https://www.skyscanner.net/*')
config.set('content.javascript.enabled', True, 'https://*.flyscoot.com/*')
config.set('content.javascript.enabled', True, 'https://*.chrono.gg/*')
config.set('content.javascript.enabled', True, 'https://*.owncube.com/*')
config.set('content.javascript.enabled', True, '*://*.learngitbranching.js.org/*')
config.set('content.javascript.enabled', True, '*://*.melpa.org/*')
config.set('content.javascript.enabled', True, '*://*.youtube.com/*')


# Enable JavaScript.
# Type: Bool
config.set('content.javascript.enabled', True, 'chrome-devtools://*')

# Enable JavaScript.
# Type: Bool
config.set('content.javascript.enabled', True, 'devtools://*')

# Enable JavaScript.
# Type: Bool
config.set('content.javascript.enabled', True, 'chrome://*/*')

# Enable JavaScript.
# Type: Bool
config.set('content.javascript.enabled', True, 'qute://*/*')

# Enable support for HTML 5 local storage and Web SQL.
# Type: Bool
c.content.local_storage = True

config.set('content.local_storage', True, 'https://*.ylab.cn/*')
config.set('content.local_storage', True, 'https://*.ficbook.net/*')
config.set('content.local_storage', True, 'https://*.author.today/*')
config.set('content.local_storage', True, 'https://fanyi.baidu.com/*')
config.set('content.local_storage', True, 'https://*.bilibili.com/*')
config.set('content.local_storage', True, 'https://*.chrono.gg/*')
config.set('content.local_storage', True, 'https://*.reddit.com/*')
config.set('content.local_storage', True, 'https://*.wechat.com/*')
config.set('content.local_storage', True, 'https://*.owncube.com/*')

# Enable WebGL.
# Type: Bool
c.content.webgl = False

# Monitor load requests for cross-site scripting attempts. Suspicious
# scripts will be blocked and reported in the inspector's JavaScript
# console. Note that bypasses for the XSS auditor are widely known and
# it can be abused for cross-site info leaks in some scenarios, see:
# https://www.chromium.org/developers/design-documents/xss-auditor
# Type: Bool
c.content.xss_auditing = True

User agent

# User agent to send.  The following placeholders are defined:  *
# `{os_info}`: Something like "X11; Linux x86_64". * `{webkit_version}`:
# The underlying WebKit version (set to a fixed value   with
# QtWebEngine). * `{qt_key}`: "Qt" for QtWebKit, "QtWebEngine" for
# QtWebEngine. * `{qt_version}`: The underlying Qt version. *
# `{upstream_browser_key}`: "Version" for QtWebKit, "Chrome" for
# QtWebEngine. * `{upstream_browser_version}`: The corresponding
# Safari/Chrome version. * `{qutebrowser_version}`: The currently
# running qutebrowser version.  The default value is equal to the
# unchanged user agent of QtWebKit/QtWebEngine.  Note that the value
# read from JavaScript is always the global value.
# Type: FormatString

# config.set('content.headers.user_agent', 'Mozilla/5.0 ({os_info}) AppleWebKit/{webkit_version} (KHTML, like Gecko) {upstream_browser_key}/{upstream_browser_version} Safari/{webkit_version}', 'https://web.whatsapp.com/')
# config.set('content.headers.user_agent', 'Mozilla/5.0 ({os_info}; rv:71.0) Gecko/20100101 Firefox/71.0', 'https://accounts.google.com/*')
# config.set('content.headers.user_agent', 'Mozilla/5.0 ({os_info}) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99 Safari/537.36', 'https://*.slack.com/*')
# config.set('content.headers.user_agent', 'Mozilla/5.0 ({os_info}; rv:71.0) Gecko/20100101 Firefox/71.0', 'https://docs.google.com/*')

# config.set('content.headers.user_agent', 'Mozilla/5.0 ({os_info}; rv:71.0) Gecko/20100101 Firefox/71.0')

Custom host blocking

Note that I need to set blocking method to “both” when using adblock.

c.content.blocking.hosts.lists = ["https://raw.githubusercontent.com/StevenBlack/hosts/master/hosts", "file:///home/yantar92/.config/qutebrowser/blocked-hosts"]
c.content.blocking.method = "both"
c.content.blocking.adblock.lists = ['https://easylist.to/easylist/easylist.txt', 'https://secure.fanboy.co.nz/fanboy-cookiemonster.txt']

Completion and history

# Number of commands to save in the command history. 0: no history / -1:
# unlimited
# Type: Int
c.completion.cmd_history_max_items = -1

# Height (in pixels or as percentage of the window) of the completion.
# Type: PercOrInt
c.completion.height = '30%'

# Number of URLs to show in the web history. 0: no history / -1:
# unlimited
# Type: Int
c.completion.web_history.max_items = 20000

Command loop

# Default program used to open downloads. If null, the default internal
# handler is used. Any `{}` in the string will be expanded to the
# filename, else the filename will be appended.
# Type: String
c.downloads.open_dispatcher = 'xdg-open {}'
# Editor (and arguments) to use for the `open-editor` command. The
# following placeholders are defined:  * `{file}`: Filename of the file
# to be edited. * `{line}`: Line in which the caret is found in the
# text. * `{column}`: Column in which the caret is found in the text. *
# `{line0}`: Same as `{line}`, but starting from index 0. * `{column0}`:
# Same as `{column}`, but starting from index 0.
# Type: ShellCommand
c.editor.command = ['emacsclient', '-c', '{}']

# Automatically enter insert mode if an editable element is focused
# after loading the page.
# Type: Bool
c.input.insert_mode.auto_load = False

# Leave insert mode if a non-editable element is clicked.
# Type: Bool
c.input.insert_mode.auto_leave = False

# Switch to insert mode when clicking flash and other plugins.
# Type: Bool
c.input.insert_mode.plugins = True

# Leave insert mode when starting a new page load. Patterns may be
# unreliable on this setting, and they may match the url you are
# navigating to, or the URL you are navigating from.
# Type: Bool
c.input.insert_mode.leave_on_load = False

Key bindings

# Translate russian keys to English bindings
# Source: https://www.reddit.com/r/qutebrowser/comments/h85sk6/hotkeys_and_different_alphabets/
c.bindings.key_mappings = {
    "<Ctrl-6>": "<Ctrl-^>",
    "<Ctrl-Enter>": "<Ctrl-Return>",
    "<Ctrl-I>": "<Tab>",
    "<Ctrl-J>": "<Return>",
    "<Ctrl-M>": "<Return>",
    "<Ctrl-[>": "<Escape>",
    "<Enter>": "<Return>",
    "<Shift-Enter>": "<Return>",
    "<Shift-Return>": "<Return>",
    'Й': 'Q', 'й': 'q',
    'Ц': 'W', 'ц': 'w',
    'У': 'E', 'у': 'e',
    'К': 'R', 'к': 'r',
    'Е': 'T', 'е': 't',
    'Н': 'Y', 'н': 'y',
    'Г': 'U', 'г': 'u',
    'Ш': 'I', 'ш': 'i',
    'Щ': 'O', 'щ': 'o',
    'З': 'P', 'з': 'p',
    'Х': '{', 'х': '[',
    'Ъ': '}', 'ъ': ']',
    'Ф': 'A', 'ф': 'a',
    'Ы': 'S', 'ы': 's',
    'В': 'D', 'в': 'd',
    'А': 'F', 'а': 'f',
    'П': 'G', 'п': 'g',
    'Р': 'H', 'р': 'h',
    'О': 'J', 'о': 'j',
    'Л': 'K', 'л': 'k',
    'Д': 'L', 'д': 'l',
    'Ж': ':', 'ж': ';',
    'Э': '"', 'э': '\'',
    'Я': 'Z', 'я': 'z',
    'Ч': 'X', 'ч': "x",
    'С': 'C', 'с': "c",
    'М': 'V', 'м': "v",
    'И': 'B', 'и': "b",
    'Т': 'N', 'т': "n",
    'Ь': 'M', 'ь': "m",
    'Б': '<', 'б': ",",
    'Ю': '>', 'ю': ".",
    }

# Bindings for normal mode
config.bind('d', 'tab-close')
config.bind('u', 'undo')
config.bind('r', 'reload')
config.bind('<', 'spawn --userscript org-protocol-add-bookmark.sh --silent')
config.bind('<Alt+l>', 'clear-keychain ;; search ;; fullscreen --leave')
config.bind('<Alt+Д>', 'clear-keychain ;; search ;; fullscreen --leave')
config.bind('<Ctrl+Shift+m>', 'bookmark-add')
config.bind('<Ctrl+g>', 'clear-keychain ;; search ;; fullscreen --leave')
config.bind('<Ctrl+i>', 'back')
config.bind('<Ctrl+m>', 'spawn --userscript org-protocol-add-bookmark.sh --rss')
config.bind('<Ctrl+o>', 'forward')
config.unbind('<Ctrl+x>')
config.bind('<Ctrl+П>', 'clear-keychain ;; search ;; fullscreen --leave')
config.bind('<F3>', 'record-macro')
config.bind('<F4>', 'run-macro')
config.bind('<F5>', 'mode-enter set_mark')
config.bind('<F6>', 'mode-enter jump_mark')
config.bind('<Shift+Д>', 'scroll-page 0 -0.5')
config.bind('<Shift+Л>', 'tab-prev')
config.bind('<Shift+М><Shift+Щ>', 'spawn --userscript  mpv-qutebrowser.sh {url}')
config.bind('<Shift+М><Щ>', 'hint links spawn --userscript  mpv-qutebrowser.sh {hint-url}')
config.bind('<Shift+О>', 'tab-next')
config.bind('<Shift+Р>', 'hint all tab')
config.bind('<Д>', 'scroll-page 0 0.5')
config.bind('<Ж><И>', 'hint all tab-bg')
config.bind('<Л>', 'scroll up')
config.bind('<М><Щ>', 'hint links spawn --userscript mpv-qutebrowser.sh {hint-url}')
config.bind('<О>', 'scroll down')
config.bind('<Щ>', 'cmd-set-text -s :open')
config.unbind('F')
config.bind('H', 'hint all tab')
config.bind('L', 'scroll-page 0 -0.5')
config.bind('M', 'spawn --userscript org-protocol-add-bookmark.sh')
config.bind('O', 'cmd-set-text -s :open -t')
config.bind('vO', 'spawn --userscript  mpv-qutebrowser.sh {url}')
config.bind('vd', 'cmd-set-text -s :download-open')
config.bind('vo', 'hint links spawn --userscript  mpv-qutebrowser.sh {hint-url}')
config.unbind('f')
config.bind('fM', 'spawn --userscript org-protocol-add-bookmark-force.sh')
config.bind('gL', 'open https://outline.com/{url}')
config.bind('h', 'hint')
config.bind(';b', 'hint links tab-bg --rapid')
config.bind('<Alt+h>', 'hint all hover')
config.bind('l', 'scroll-page 0 0.5')
config.bind('o', 'cmd-set-text -s :open')
config.unbind('q')
config.unbind('v')
config.bind('vd', 'download-open')
config.bind('vi', 'mode-enter caret')
config.bind('vo', 'hint links spawn --userscript mpv-qutebrowser.sh {hint-url}')

config.bind(';D', 'hint all delete')

# Bindings for caret mode
config.bind('<Alt+l>', 'mode-leave', mode='caret')
config.bind('<Ctrl+g>', 'mode-leave', mode='caret')
config.bind('<Ctrl+П>', 'mode-leave', mode='caret')
config.bind('D', 'yank selection', mode='caret')
config.bind('I', 'move-to-prev-word', mode='caret')
config.bind('O', 'move-to-next-word', mode='caret')
config.bind('P', 'scroll right', mode='caret')
config.bind('U', 'scroll left', mode='caret')
config.bind('[', 'move-to-start-of-prev-block', mode='caret')
config.bind(']', 'move-to-start-of-next-block', mode='caret')
config.bind('i', 'move-to-prev-char', mode='caret')
config.bind('n', 'reverse-selection', mode='caret')
config.bind('o', 'move-to-next-char', mode='caret')
config.bind('p', 'move-to-end-of-line', mode='caret')
config.bind('u', 'move-to-start-of-line', mode='caret')
config.bind('{', 'move-to-end-of-prev-block', mode='caret')
config.bind('}', 'move-to-end-of-next-block', mode='caret')

# Bindings for command mode
config.bind('<Alt+j>', 'completion-item-focus next', mode='command')
config.bind('<Alt+k>', 'completion-item-focus prev', mode='command')
config.bind('<Alt+l>', 'mode-leave', mode='command')
config.bind('<Ctrl+Alt+h>', 'rl-backward-kill-word', mode='command')
config.bind('<Ctrl+Shift+o>', 'completion-item-focus prev-category', mode='command')
config.bind('<Ctrl+g>', 'mode-leave', mode='command')
config.bind('<Ctrl+o>', 'completion-item-focus next-category', mode='command')
config.bind('<Ctrl+П>', 'mode-leave', mode='command')

# Bindings for hint mode
config.bind('<Alt+l>', 'mode-leave', mode='hint')
config.bind('<Ctrl+g>', 'mode-leave', mode='hint')
config.bind('<Ctrl+П>', 'mode-leave', mode='hint')


# Bindings for insert mode
config.bind('<Alt+l>', 'mode-leave', mode='insert')
config.bind('<Ctrl+Alt+h>', 'fake-key <Shift-Left> ;; fake-key <Backspace>', mode='insert')
config.bind('<Ctrl+g>', 'mode-leave', mode='insert')
config.bind('<Ctrl+h>', 'fake-key <Delete>', mode='insert')
config.bind('<Ctrl+y>', 'insert-text {clipboard}', mode='insert')
config.bind('<Ctrl+П>', 'mode-leave', mode='insert')
config.bind('<Shift+Ins>', 'insert-text {primary}', mode='insert')

# Bindings for passthrough mode
config.bind('<Alt+l>', 'mode-leave', mode='passthrough')
config.bind('<Ctrl+g>', 'mode-leave', mode='passthrough')
config.bind('<Ctrl+y>', 'insert-text {clipboard}', mode='passthrough')
config.bind('<Ctrl+П>', 'mode-leave', mode='passthrough')
config.bind('<Shift+Ins>', 'insert-text {primary}', mode='passthrough')

# Bindings for prompt mode
config.bind('<Alt+l>', 'mode-leave', mode='prompt')
config.bind('<Ctrl+Alt+h>', 'rl-backward-kill-word', mode='prompt')
config.bind('<Ctrl+d>', 'rl-delete-char', mode='prompt')
config.bind('<Ctrl+g>', 'mode-leave', mode='prompt')
config.bind('<Ctrl+П>', 'mode-leave', mode='prompt')

# Bindings for register mode
config.bind('<Alt+l>', 'mode-leave', mode='register')
config.bind('<Ctrl+g>', 'mode-leave', mode='register')
config.bind('<Ctrl+П>', 'mode-leave', mode='register')

#hyposes.is
config.bind('A', "jseval (function(){window.hypothesisConfig=function(){return{showHighlights:true,appType:'bookmarklet'};};var d=document,s=d.createElement('script');s.setAttribute('src','https://localhost:3001/hypothesis');d.body.appendChild(s)})();")

High DPI display

c.qt.highdpi = True

Bash script opening a new tab in qutebrowser depending on the url

Use mpv for videos or Archivebox for locally archived URLs. The script uses scripts defined in Archivebox.

URL="$1"

case "$URL" in
    *ted.com/talks*|*bilibili.com*|*youtube.com*)
	mpv "$URL" && exit
	;;
esac

grep "$URL" ~/.data/web-mirror/sources/* >/dev/null 2>&1 &&\
    ARCHIVE_ID=$(grep "$URL" ~/.data/web-mirror/sources/* | cut -f1 -d: | cut -d- -f2 | cut -d/ -f3) &&\
    ARCHIVE_FILE=$(find ~/.data/web-mirror/archive/ | grep $ARCHIVE_ID | grep singlefile.html | tail -n1)

if [[ ! -z "$ARCHIVE_FILE" ]]; then
    URL="${ARCHIVE_FILE}"
fi

# # fi
# echo "local awful = require('awful'); local screen = awful.screen.focused(); local tag = screen.tags[2]  if tag then tag:view_only() end" | awesome-client

#from https://github.com/qutebrowser/qutebrowser/blob/master/scripts/open_url_in_instance.sh
_url="$URL"

qutebrowser ":open -t ${_url}"

# emacsclient -c -e "(eaf-open-browser \"$URL\")"

Adblocking

In Qutebrowser 2.0.0, there is better adblocking using python adblocking library [[notmuch:id:20210128104817.ljsjkjoxtp73e3k7@aragog.localdomain][Email from Florian Bruhin via qutebrowser-announce: [qutebrowser-announce] qutebrowser v2.0.0 released (with better adblocker)!]]

Need to install dev-python/adblock from 2xsaiko overlay.

Torbrowser

app-eselect/eselect-repository

Need to use torbrowser overlay

eselect repository enable torbrowser
www-client/torbrowser-launcher
*/*::torbrowser
www-client/torbrowser-launcher::torbrowser
www-client/torbrowser-launcher::torbrowser ~amd64

Download manager (including torrent): Aria2

net-misc/aria2
net-misc/aria2 bittorrent

Emacs

app-editors/emacs:30-vcs
app-editors/emacs:29
app-editors/emacs:28
app-editors/emacs:27
app-editors/emacs:26

Use latest master

app-editors/emacs **
app-editors/emacs cairo dbus dynamic-loading gif jpeg libxml2 png sound source svg tiff wide-int xft xwidgets harfbuzz json gui native-comp imagemagick
# required for pdf-tools
app-text/poppler cairo
#support webp
media-gfx/imagemagick jpeg tiff webp png svg xml wmf raw
#native-comp
sys-devel/gcc jit

Allow pdf->png conversion using imagemagick. Required for org-mode preview working properly. [2023-01-10 Tue] Does not work - changes in sed??

sed -i  's|  <!-- <policy domain="module" rights="none" pattern="{PS,PDF,XPS}" /> -->|  <policy domain="module" rights="read|write" pattern="{PS,PDF,XPS}" />|' policy.xml
sed -i  's|<policy domain="coder" rights="none" pattern="PDF" />|<!-- <policy domain="coder" rights="none" pattern="PDF" /> -->|' policy.xml

Install mime handler for Emacs

[Desktop Entry]
Type=Application
Version=1.0
Name=Emacs Client
Exec=emacsclient -c %f
Icon=emacs-icon
Terminal=false
MimeType=text/css;text/english;text/html;text/plain;text/x-c;text/x-chdr;text/x-csrc;text/x-c++;text/x-c++hdr;text/x-c++src;text/x-java;text/x-makefile;text/x-moc;text/x-pascal;text/x-tcl;text/x-tex;application/x-shellscript;application/pdf;inode/directory;inode/mount-point;image/png;image/tiff;

Note: I need to update desktop database upon changing this file

update-desktop-database ~/.local/share/applications

Required fonts

source code pro

media-fonts/source-code-pro

symbola

eselect repository enable guru

Do not enable any other packages from the overlay

*/*::guru
media-fonts/symbola::guru
media-fonts/symbola
media-fonts/symbola UFAS

FontAwesome

media-fonts/fontawesome

Noto

media-fonts/noto

Noto emoji, required to get colored emojis in Emacs

media-fonts/noto-emoji

Sarasa Gothic (for Russian monospaced text)

[[id:21f06900e232809e79ecdd07e0d84647f30e8975][be5invis [Github] Sarasa-Gothic: Sarasa Gothic / 更纱黑体 / 更紗黑體 / 更紗ゴシック / 사라사 고딕]]

eselect repository enable gentoo-zh

Do not enable any other packages from the overlay

*/*::gentoo-zh
media-fonts/sarasa-gothic::gentoo-zh
media-fonts/sarasa-gothic

all-the-icons

Need to run elisp:all-the-icons-install-fonts

Clocking management scripts

emacsclient -e "(yant/punch-in-organization)"
emacsclient -e "(yant/punch-in-home)"
emacsclient -e "(yant/punch-out)"

System wibar indicator showing currently clocked in tasks

Works together with Externally controlled widgets (awesome WM) The information about time balance multiplier is shown in other script

add links to balance multiplier and the balance scriptEND
while true; do org-clocked-in.sh 2>> ~/.log/org-clocked-in.log; sleep 5; done &
# Make sure it exists
[[ -f "/home/yantar92/.org-clock-in" ]] || touch ~/.org-clock-in
BALANCE=""
[[ "$1" == "balance" ]] && BALANCE="1"

PID="$(pgrep awesome | head -n1)"
[[ -z "$PID" && -d "/proc/$PID" ]] || export DBUS_SESSION_BUS_ADDRESS="$(grep -z DBUS_SESSION_BUS_ADDRESS /proc/$PID/environ | tr '\0' '\n' | cut -d= -f2-)"


if [[ "$(cat ~/.org-clock-in | head -n1)" =~ "No clocked in task" ]]; then
    echo "orgstatus:set_text(\"No clocked in task\")" | awesome-client
    if [[ ! -z "$BALANCE" ]]; then
	BAL=$(cat ~/.org-clock-in  | head -n2 | tail -n1 | cut -d. -f1)
        BAL=$(echo "$BAL/60" | bc -l | sed -r 's/([^.]*\..).*/\1/')
        LOSSAGE=$(cat ~/.org-clock-in  | tail -n1 | cut -d. -f1)
        LOSSAGE=$(echo "$LOSSAGE/-60" | bc -l | sed -r 's/([^.]*\..).*/\1/')
        [[ "$(echo "$LOSSAGE < 0" | bc -l)" == "1" ]] && LOSSAGE="<span color=\"#FF0000\">⏺</span>" || LOSSAGE="<span color=\"#00FF00\">⏺</span>"
	[[ "$(echo "$BAL < 0" | bc -l)" == "1" ]] && BAL_STRING="<span color=\"#FF0000\">${BAL}h</span>" \
		|| BAL_STRING="${BAL}h"
        BAL_STRING="${BAL_STRING}<span font=\"12\">${LOSSAGE}</span>"
        echo $BAL_STRING
    fi
    exit
fi

time_string="`cat ~/.org-clock-in | cut -d[ -f2 | cut -d] -f1 | head -n1`"
time_before="`echo $time_string | cut -dh -f1`:`echo $time_string | sed -r 's/[0-9]+h //' | cut -dm -f1`"
if [[ "$time_string" =~ .+/.+ ]]; then
    effort_estimate="/`echo $time_string | sed -r 's/.+\///' | sed 's/ \([0-9]min\)/ 0\1/' | sed -r 's/h /:/' | sed -r 's/min//'`"
fi

# cat ~/.org-clock-in | cut -d[ -f2 | cut -d] -f1 | head -n1 | cut -d'(' -f1 | grep / &&\
     #     estimate="/`cat ~/.org-clock-in | cut -d[ -f2 | cut -d] -f1 | head -n1 | cut -d'(' -f1 | sed -r 's|.+/||'`"\
     # 	|| estimate=""

 time_plus="`timefromchange.sh ~/.org-clock-in | grep -Eo "[0-9]+ hours" | grep -Eo "[0-9]+" || echo 0`:`timefromchange.sh ~/.org-clock-in | grep -Eo "[0-9]+ minutes" | grep -Eo "[0-9]+" || echo 0`"
 hours_before="`echo $time_before | cut -d':' -f1 | sed -r 's/^0*([0-9]+)/\1/'`"
 hours_plus="`echo $time_plus | cut -d':' -f1 | sed -r 's/^0*([0-9]+)/\1/'`"
 minutes_before="`echo $time_before | cut -d':' -f2 | sed -r 's/^0*([0-9]+)/\1/'`"
 minutes_plus="`echo $time_plus | cut -d':' -f2 | sed -r 's/^0*([0-9]+)/\1/'`"
   hours_now="$(($hours_before + $hours_plus))"
   minutes_now="$(($minutes_before + $minutes_plus))"

   while (($minutes_now >= 60)); do
       minutes_now="$(($minutes_now - 60))";
       hours_now="$(($hours_now + 1))";
   done

   if (($minutes_now < 10)); then
       minutes_now="0$minutes_now";
   fi

time_now="$hours_now:$minutes_now"
RES="[ ${time_now}${effort_estimate} ] :: `cat ~/.org-clock-in | cut -d] -f 2- | head -n1`"
RES="$(echo "$RES" | sed -r 's/\((.*)\)/\1/')"

RES=$(echo "$RES" | sed -r "s/'/\"/g") # fix "'"

echo "orgstatus:set_text('$RES')" | awesome-client 2>&1 >>/dev/null;

if [[ ! -z "$BALANCE" ]]; then
    MULT="$(cat ~/.org-clock-in | tail -n2 | head -n1)";
    BAL="$(cat ~/.org-clock-in | tail -n3 | head -n1)";
    BAL_STRING=$(echo "($hours_plus * $MULT * 60 + $minutes_plus * $MULT + $BAL)/60.0" | bc -l | sed -r 's/([^.]*\..).*/\1/');
    LOSSAGE=$(cat ~/.org-clock-in  | tail -n1 | cut -d. -f1)
    LOSSAGE=$(echo "$LOSSAGE/-60" | bc -l | sed -r 's/([^.]*\..).*/\1/')
    [[ "$(echo "$LOSSAGE < 0" | bc -l)" == "1" ]] && LOSSAGE="<span color=\"#FF0000\">⏺</span>" || LOSSAGE="<span color=\"#00FF00\">⏺</span>"
    if [[ $(echo "$BAL_STRING < 0" | bc -l) == "1" ]]; then
	BAL_STRING="<span color=\"#FF0000\">${BAL_STRING}h</span>"
        [[ ! -f /tmp/negative-time-balance ]] && notify-send -u critical "Negative time balance"
        touch /tmp/negative-time-balance
    else
	BAL_STRING="${BAL_STRING}h"
        rm -f /tmp/negative-time-balance
    fi
    [[ $(echo "$MULT < 0" | bc -l) == "1" ]] && BAL_STRING="${BAL_STRING}<span color=\"#FF0000\">⇓"\
	    || BAL_STRING="${BAL_STRING}<span color=\"#00FF00\">⇑"
    BAL_STRING="${BAL_STRING}<span font=\"8\">$(echo $MULT | sed -r 's/-//')x</span></span><span font=\"12\">${LOSSAGE}</span>"
    echo $BAL_STRING
fi

Helper script calculating time from last change of a file

displaytime.sh $(expr $(date +%s) - $(date +%s -r $1))
function displaytime {
  local T=$1
  local D=$((T/60/60/24))
  local H=$((T/60/60%24))
  local M=$((T/60%60))
  local S=$((T%60))
  [[ $D > 0 ]] && printf '%d days ' $D
  [[ $H > 0 ]] && printf '%d hours ' $H
  [[ $M > 0 ]] && printf '%d minutes ' $M
  [[ $D > 0 || $H > 0 || $M > 0 ]] && printf 'and '
  printf '%d seconds\n' $S
}

displaytime $1

Spell checking (aspell)

Dictionaries

  • app-dicts/aspell-en
app-text/aspell
app-text/hunspell

Spell checking enchant (for Emacs =jinx= - spell checker)

app-text/enchant

w3m browser for rendering

www-client/w3m

Langdetect python library to translate elfeed entries

Automatically translate non-English and non-Russian/Ukrainian titles.

dev-python/langdetect::guru
dev-python/langdetect
dev-python/langdetect python3_10

xprintidle to allow Org auto-clock-out on X11 idle

x11-misc/xprintidle

Testing emacs repo

function yes_or_no {
    while true; do
        read -p "$* [y/n]: " yn
        case $yn in
            [Yy]*) return 0  ;;  
	    [Nn]*) echo "Aborted" ; return  1 ;;
	esac
done
}

set -e
make cleanall
make EMACS=emacs-26 $* test || (echo "Failed to run tests using $(emacs-26 --version | head -n1)"; yes_or_no " Continue?")
make cleanall
make EMACS=emacs-27 $* test || (echo "Failed to run tests using $(emacs-27 --version | head -n1)"; yes_or_no " Continue?")
make cleanall
make EMACS=emacs-28 $* test || (echo "Failed to run tests using $(emacs-28-vcs --version | head -n1)"; yes_or_no " Continue?")
make cleanall
make EMACS=emacs-29-vcs $* test || (echo "Failed to run tests using $(emacs-29-vcs --version | head -n1)"; yes_or_no " Continue?")
make cleanall
LANG="C" make EMACS=emacs-29-vcs $* test || (echo "Failed to run tests using LANG=C $(emacs-29-vcs --version | head -n1)"; yes_or_no " Continue?")
make cleanall
LANG="de_DE.UTF-8" make EMACS=emacs-29-vcs $* test || echo "Failed to run tests using LANG=de_DE.UTF-8 $(emacs-29-vcs --version | head -n1)"

Silver searcher (ag)

sys-apps/the_silver_searcher

Ledger

app-office/ledger
app-office/ledger doc

Default args

yant/ledger-file
--pedantic
--explicit
--file <<ledger-file()>>
--effective

Weekly expenses script

_LAST_YEAR=$(echo $(date +%Y) - 2 | bc -l)
BUDGET_CUTOFF="from ${_LAST_YEAR}-$(date +%m)"
ledger --no-pager -f <<ledger-file()>>  reg -X <<ledger-main-currency()>> Expenses and not Work and not Gifts and not Medical -W -n -A -b "${BUDGET_CUTOFF}"
_LAST_YEAR=$(echo $(date +%Y) - 2 | bc -l)
BUDGET_CUTOFF="from ${_LAST_YEAR}-$(date +%m)"
ledger --no-pager -f <<ledger-file()>>  reg -X <<ledger-main-currency()>> Income -W -n -A -b "${BUDGET_CUTOFF}"
_LAST_YEAR=$(echo $(date +%Y) - 2 | bc -l)
BUDGET_CUTOFF="from ${_LAST_YEAR}-$(date +%m)"
ledger --no-pager -f <<ledger-file()>>  reg -X $ Expenses and not Work and not Gifts and not Medical -W -n -A -b "${BUDGET_CUTOFF}"
_LAST_YEAR=$(echo $(date +%Y) - 2 | bc -l)
BUDGET_CUTOFF="from ${_LAST_YEAR}-$(date +%m)"
ledger --no-pager -f <<ledger-file()>>  reg -X EUR Expenses and not Work and not Gifts and not Medical -W -n -A -b "${BUDGET_CUTOFF}"

Food status script

ledger --no-pager -f <<ledger-file()>>  reg --budget -X <<ledger-main-currency()>> Food  -U -p "until tomorrow"

Continuous monitoring of budget

It is very useful to keep track of the important budget variables regularly [[id:TEDbelle2021what_your_money_habit_reveal_about_youee4][Robert A. Belle [TED] (2021) What your money habits reveal about you]].

I have an Awesome WM widget showing the most important budgets on screen (seeing things is important, see [[id:6231301de4667cef9614b4e945cf417b35b08ced][James Clear [Amazon] (2018) Atomic Habits, Chapter 6]]).

The widget is populated using the following script. The script assumes Awesome WM markup convention.

The script also shows my time budget.

yant/ledger-main-currency
yant/ledger-main-currency-symbol
yant/ledger-food-currency
yant/ledger-food-currency-symbol
yant/ledger-max-balance-food
yant/ledger-min-balance-food
yant/ledger-normal-balance-food
yant/ledger-emergency-fund-months
yant/min-weekly-spending
yant/ledger-checking-account

The idea of the script is regular reminder to not spend too much money on certain things and not forget spending enough money on others:

  • Food (should not overspend)
  • Emergency fund
  • Small things that improve well-being, but are not strictly necessary (I keep forgetting to buy those even if I know what can make my life easier)
  • Leisure (I usually go from one extreme to other: either not spend much or vice versa)
  • Retirement fund
  • Reminder to invest extra money
note how I calculate retirement accountEND
# export DBUS_SESSION_BUS_ADDRESS environment variable
PID=$(pgrep awesome | head -n1)
[[ -z "$PID" && -d "/proc/$PID" ]] && export DBUS_SESSION_BUS_ADDRESS="$(grep -z DBUS_SESSION_BUS_ADDRESS /proc/$PID/environ | tr '\0' '\n' | cut -d= -f2-)"

UNCONVERTED_CURRENCIES="$(ledger -X  <<ledger-main-currency()>> bal Expenses | sed -E '/<<ledger-main-currency()>>|----+/d')"

_LAST_YEAR=$(echo $(date +%Y) - 2 | bc -l)
BUDGET_CUTOFF="from ${_LAST_YEAR}-$(date +%m)"

weekly_min_spending="<<min-weekly-spending()>>"

if [[ ! -z "$UNCONVERTED_CURRENCIES" ]]; then
    [[ "$PROJECTING" == "1" ]] && echo "balance.markup=''" | awesome-client || echo "balance.markup='༼ <span color=\"#FF0000\"><b>Ledger error</b></span>༼ <span color=\"#FF0000\">Not all currencies convertable</span>༼ '" | awesome-client;
    exit;
fi

BAL_RENT="$(ledger bal --budget -X <<ledger-main-currency()>> Expenses:Personal:Rental -p 'until tomorrow' | awk '{print $1}' | sed -r 's/[^0-9-]+//'  | sed -r 's/,//g')";
[[ -z "$BAL_RENT" ]] && BAL_RENT="0"
BAL_RENT="$(echo "${BAL_RENT}*(-1)" | bc -l)";
under_0=$(echo "$BAL_RENT < 0" | bc -l)
if [[ "$under_0" == "1" ]]; then
    BAL_RENT="<span color=\"#FF0000\">⏺</span>"
else
    BAL_RENT="<span color=\"#00FF00\">⏺</span>"
fi


BAL_ENTERTAINMENT="$(ledger bal --budget -X <<ledger-main-currency()>> Entertainment  -p 'until tomorrow' | awk '{print $1}' | sed -r 's/[^0-9-]+//'  | tail -n1 | sed -r 's/,//g')";
TODAY_BAL_FOOD="$(ledger bal -X <<ledger-food-currency()>> -b `date +%F` Food | tail -n1 | sed -r 's/[^0-9-]+//'  | cut -d' ' -f 1 | sed -r 's/,//g')"
[[ -z "$TODAY_BAL_FOOD" ]] && TODAY_BAL_FOOD="0"
MAX_BAL_FOOD="<<max-balance-food()>>"
NORMAL_BAL_FOOD="<<normal-balance-food()>>"
MIN_BAL_FOOD="<<min-balance-food()>>"
under_limit="0"

BAL_FOOD="$(ledger bal --budget -X <<ledger-food-currency()>> Food -p 'until tomorrow' | awk '{print $1}' | sed -r 's/[^0-9-]+//' | tail -n1 | sed -r 's/,//g')";
if [[ -n "$BAL_FOOD" ]]; then
    BAL_FOOD="$(echo "${BAL_FOOD}*(-1)" | bc -l)";
    res="$(echo "$BAL_FOOD + $TODAY_BAL_FOOD > $MAX_BAL_FOOD" | bc -l)";
    [[ "$res" == "1" ]] && BAL_FOOD=$(echo "$MAX_BAL_FOOD - $TODAY_BAL_FOOD" | bc -l);
    under_limit=$(echo "$BAL_FOOD < $MIN_BAL_FOOD" | bc -l);
    [[ "$under_limit" == "1" ]] && BAL_FOOD="$MIN_BAL_FOOD";
else
    BAL_FOOD="N/A";
fi
under_limit=$(echo "$BAL_FOOD < $NORMAL_BAL_FOOD" | bc -l)
under_0=$(echo "$BAL_FOOD < 0" | bc -l)
BAL_FOOD=$(echo "$BAL_FOOD" | sed -r 's/^\./0./')
BAL_FOOD=$(echo "$BAL_FOOD" | sed -r 's/^-\./-0./')
if [[ "$under_0" == "1" ]]; then
    BAL_FOOD="<span color=\"#FF0000\"><<ledger-food-currency-symbol()>>$BAL_FOOD milk\!</span>"
elif [[ "$under_limit" == "1" ]]; then
    BAL_FOOD="<span color=\"#FF0000\"><<ledger-food-currency-symbol()>>$BAL_FOOD canteen\!</span>"
else
    BAL_FOOD="<<ledger-food-currency-symbol()>>$BAL_FOOD"
fi

# BAL_ENTERTAINMENT_SHORT="$(echo "$BAL_ENTERTAINMENT/1000.0*(-1)" | bc -l | grep -oE '^-?[0-9]*\.?[0-9]{0,2}')k"
# BAL_ENTERTAINMENT_SHORT="$(echo $BAL_ENTERTAINMENT_SHORT | sed -r 's/^(-?)\./\10\./')"
BAL_ENTERTAINMENT_SHORT="$(echo "$BAL_ENTERTAINMENT*(-1)" | bc -l | grep -oE '^-?[0-9]*')"
[[ "$(echo "$BAL_ENTERTAINMENT > 0" | bc -l)" == "1" ]] && BAL_ENTERTAINMENT_SHORT="<span color=\"#FF0000\"><<ledger-main-currency-symbol()>>$BAL_ENTERTAINMENT_SHORT</span>" || BAL_ENTERTAINMENT_SHORT="<<ledger-main-currency-symbol()>>$BAL_ENTERTAINMENT_SHORT"

WEEKLY_INCOME_AVG="$(ledger r -X <<ledger-main-currency()>> Income -W -n -A -p "${BUDGET_CUTOFF} until tomorrow" | tail -n1 | sed -r 's/.+?<<ledger-main-currency()>>//' | sed -r 's/^-//' | sed -r 's/,//')" # average weekly income from the beginning of the year
[[ -z "$WEEKLY_INCOME_AVG" ]] && WEEKLY_INCOME_AVG="0"
WEEKLY_INCOME_AVG_NO_SALARY="$(ledger r -X <<ledger-main-currency()>> Income and not Salary -W -n -A -p "${BUDGET_CUTOFF} until tomorrow" | tail -n1 | sed -r 's/.+?<<ledger-main-currency()>>//' | sed -r 's/^-//' | sed -r 's/,//')" # average weekly income minus salary
[[ -z "$WEEKLY_INCOME_AVG_NO_SALARY" ]] && WEEKLY_INCOME_AVG_NO_SALARY="0"
WEEKLY_INCOME_AVG_NO_SALARY_NO_GIFTS="$(ledger r -X <<ledger-main-currency()>> Income and not Salary and not Gifts -W -n -A -p "${BUDGET_CUTOFF} until tomorrow" | tail -n1 | sed -r 's/.+?<<ledger-main-currency()>>//' | sed -r 's/^-//' | sed -r 's/,//')" # average weekly income minus salary
[[ -z "$WEEKLY_INCOME_AVG_NO_SALARY_NO_GIFTS" ]] && WEEKLY_INCOME_AVG_NO_SALARY_NO_GIFTS="0"


MONTHLY_INCOME_AVG="$(echo "52.178574 / 12.0 * $WEEKLY_INCOME_AVG" | bc -l)"
# https://www.wisebread.com/figuring-the-size-of-your-emergency-fund
WEEKLY_SPENDING_AVG="$(ledger r -X <<ledger-main-currency()>> Expenses and not Work and not Gifts -W -n -A -p "${BUDGET_CUTOFF} until tomorrow" | tail -n1 | sed -r 's/.+?<<ledger-main-currency()>>//' | sed -r 's/,//')" # average expenses from beginning last year in <<ledger-main-currency()>>
[[ -z "$WEEKLY_SPENDING_AVG" ]] && WEEKLY_SPENDING_AVG="0"
under_min_spending=$(echo "$WEEKLY_SPENDING_AVG < $weekly_min_spending" | bc -l)
[[ "$under_min_spending" == "1" ]] && WEEKLY_SPENDING_AVG="$weekly_min_spending"
MONTHLY_SPENDING_AVG="$(echo "52.178574 / 12.0 * $WEEKLY_SPENDING_AVG" | bc -l)"

INCOME_RATIO="$(echo "100.0 * $WEEKLY_INCOME_AVG / $WEEKLY_SPENDING_AVG" | bc -l | sed -r 's/\..*//')"
[[ -z "$INCOME_RATIO" ]] && INCOME_RATIO="0"
INCOME_RATIO_NO_SALARY="$(echo "100.0 * $WEEKLY_INCOME_AVG_NO_SALARY / $WEEKLY_SPENDING_AVG" | bc -l | sed -r 's/\..*//')"
[[ -z "$INCOME_RATIO_NO_SALARY" ]] && INCOME_RATIO_NO_SALARY="0"
INCOME_RATIO_NO_SALARY_NO_GIFTS="$(echo "100.0 * $WEEKLY_INCOME_AVG_NO_SALARY_NO_GIFTS / $WEEKLY_SPENDING_AVG" | bc -l | sed -r 's/\..*//')"
[[ -z "$INCOME_RATIO_NO_SALARY_NO_GIFTS" ]] && INCOME_RATIO_NO_SALARY_NO_GIFTS="0"

# args: INCOME_RATIO
colorize_ratio () {
    local INCOME_RATIO="$1"
    if [[ "1" == "$(echo "$INCOME_RATIO < 100" | bc -l)" ]]; then
	INCOME_RATIO="<span color=\"#FF0000\">"$INCOME_RATIO"%</span>"
    else
	if [[ "1" == "$(echo "$INCOME_RATIO < 200" | bc -l)" ]]; then
	    INCOME_RATIO="<span color=\"#FFFF00\">"$INCOME_RATIO"%</span>"
	else
	    INCOME_RATIO="<span color=\"#00FF00\">"$INCOME_RATIO"%</span>"
	fi
    fi
    echo "$INCOME_RATIO"
}

INCOME_RATIO=$(colorize_ratio "$INCOME_RATIO")
INCOME_RATIO_NO_SALARY=$(colorize_ratio "$INCOME_RATIO_NO_SALARY")
INCOME_RATIO_NO_SALARY_NO_GIFTS=$(colorize_ratio "$INCOME_RATIO_NO_SALARY_NO_GIFTS")

EMERGENCY_FUND="$(echo "$MONTHLY_SPENDING_AVG * <<emergency-fund-months()>>" | bc -l)"
BAL_EMERGENCY="$(ledger bal -X <<ledger-main-currency()>> Assets:Checking -p 'until tomorrow' | awk '{print $1}' | sed -r 's/<<ledger-main-currency()>>//' | sed -r 's/,//g' | tail -n1)"
 if [[ "1" == "$(echo "$EMERGENCY_FUND > $BAL_EMERGENCY" | bc -l)" ]]; then
     BAL_EMERGENCY="<span color=\"#FF0000\">$(echo "($BAL_EMERGENCY - $EMERGENCY_FUND)/$MONTHLY_INCOME_AVG" | bc -l | sed -r 's/(\..{1}).+/\1/' | sed -r 's/^(-?)(\.)/\10\2/')m</span>"
 else
     if [[ "0" == "$EMERGENCY_FUND" ]]; then
	 BAL_EMERGENCY="<span color=\"#FFFF00\">N/A</span>"
     else
	 BAL_EMERGENCY="<span color=\"#00FF00\">ok</span>"
     fi
 fi


 BAL_CHECKING="$(ledger bal -X <<ledger-main-currency()>> Assets:Checking -p 'until tomorrow' | awk '{print $1}' | sed -r 's/<<ledger-main-currency()>>//' | sed -r 's/,//g' | tail -n1)"
 BAL_ASSETS="$(ledger bal -X <<ledger-main-currency()>> Assets -p 'until tomorrow' | awk '{print $1}' | sed -r 's/<<ledger-main-currency()>>//' | sed -r 's/,//g' | tail -n1)"
 CHECKING_EXTRA="$(echo $BAL_CHECKING - ${MONTHLY_SPENDING_AVG} - ${EMERGENCY_FUND} | bc -l)" # extra one month

 if [[ "1" == "$(echo "$CHECKING_EXTRA > 0" | bc -l)" ]]; then
     if [[ "0" == "$MONTHLY_SPENDING_AVG" ]]; then
	 EXTRA="<span color=\"#FFFF00\">N/A</span>"
     else
	 if [[ "1" == "$(echo "$CHECKING_EXTRA > $MONTHLY_SPENDING_AVG" | bc -l)" ]]; then
	     EXTRA="<span color=\"#FF0000\"><<ledger-main-currency-symbol()>>$(echo "$CHECKING_EXTRA/1000.0" | bc -l | grep -oE '^-?[0-9]*\.?[0-9]{0,1}')k</span>"
	 else
	     EXTRA="<<ledger-main-currency-symbol()>>$(echo "$CHECKING_EXTRA/1000.0" | bc -l | grep -oE '^-?[0-9]*\.?[0-9]{0,1}')k"
	 fi
     fi
 else
     EXTRA="<span color=\"#00FF00\">no</span>"
 fi

 # args: BAL_PROJECTION_YEAR
 years_left () {
     local BAL_PROJECTION_YEAR="$1"
     local YEARS_LEFT=""
     if [[ "1" == "$(echo "$BAL_PROJECTION_YEAR > 0" | bc -l)" ]]; then
	 YEARS_LEFT=$(echo "$BAL_ASSETS / $BAL_PROJECTION_YEAR" | bc -l | sed -r 's/(\..{1}).+/\1/');
	 if [[ "1" == "$(echo "$YEARS_LEFT < 5" | bc -l)" ]]; then
	     YEARS_LEFT="<span font=\"12\" color=\"#FF0000\">+${YEARS_LEFT}y</span>";
         elif [[ "1" == "$(echo "$YEARS_LEFT < 15" | bc -l)" ]]; then
             YEARS_LEFT="<span font=\"12\" color=\"#FFFF00\">+${YEARS_LEFT}y</span>";
	 else
	     YEARS_LEFT="<span font=\"12\" color=\"#99FF00\">+${YEARS_LEFT}y</span>";
	 fi
     else
	 YEARS_LEFT="<span font=\"12\" color=\"#00FF00\">+∞y</span>";
     fi
     echo "$YEARS_LEFT"
 }    

 BAL_PROJECTION_YEAR=$(echo "($WEEKLY_SPENDING_AVG - $WEEKLY_INCOME_AVG) * 52" | bc -l)
 BAL_PROJECTION_YEAR_NO_SALARY=$(echo "($WEEKLY_SPENDING_AVG - $WEEKLY_INCOME_AVG_NO_SALARY) * 52" | bc -l)
 BAL_PROJECTION_YEAR_NO_SALARY_NO_GIFTS=$(echo "($WEEKLY_SPENDING_AVG - $WEEKLY_INCOME_AVG_NO_SALARY_NO_GIFTS) * 52" | bc -l)

 YEARS_LEFT="$(years_left "$BAL_PROJECTION_YEAR")"
 YEARS_LEFT_NO_SALARY="$(years_left "$BAL_PROJECTION_YEAR_NO_SALARY")"
 YEARS_LEFT_NO_SALARY_NO_GIFTS="$(years_left "$BAL_PROJECTION_YEAR_NO_SALARY_NO_GIFTS")"

 if [ "$INCOME_RATIO" == "$INCOME_RATIO_NO_SALARY" ] ; then
     INCOME_RATIO=""
     YEARS_LEFT=""
 fi
 if [ "$INCOME_RATIO_NO_SALARY" == "$INCOME_RATIO_NO_SALARY_NO_GIFTS" ] ; then
     INCOME_RATIO_NO_SALARY=""
     YEARS_LEFT_NO_SALARY=""
 fi

 
 RETIREMENT="$(ledger b -X <<ledger-main-currency()>> "Retirement Savings" -n -p 'until tomorrow' | awk '{print $1}' | sed -r 's/<<ledger-main-currency()>>//' | sed -r 's/^-//' | sed -r 's/,//g')"
 [[ -z "$RETIREMENT" ]] && RETIREMENT="0"
 ASSETS="$(ledger bal -X <<ledger-main-currency()>> Assets -n  -p 'until tomorrow' | awk '{print $1}' | sed -r 's/<<ledger-main-currency()>>//' | sed -r 's/,//g')"
 EQUITY="$(ledger bal -X <<ledger-main-currency()>> Equity -n  -p 'until tomorrow' | awk '{print $1}' | sed -r 's/<<ledger-main-currency()>>//' | sed -r 's/,//g')"
 if [[ "1" == "$(echo "$RETIREMENT > ($ASSETS + $EQUITY)" | bc -l)" ]]; then
     RETIREMENT="<span color=\"#FF0000\">$(echo "($ASSETS + $EQUITY - $RETIREMENT)/$MONTHLY_INCOME_AVG" | bc -l | sed -r 's/(\..{1}).+/\1/' | sed -r 's/^(-?)(\.)/\10\2/')m</span>"
 else
     RETIREMENT="<span color=\"#00FF00\">ok</span>"
 fi

 OTHER="$(ledger budget Expenses:Personal:Other -X <<ledger-main-currency()>> -p "until tomorrow" | tail -n1 | awk '{print $3}' | sed -r 's/<<ledger-main-currency()>>//')"
 # OTHER="$(echo "$OTHER * -1 / 1000" | bc -l)"
 # OTHER_SHORT="$(echo $OTHER | sed -r 's/(\..{1}).+/\1/' | sed -r 's/^(-?)(\.)/\10\2/')k"
 OTHER="$(echo "$OTHER * -1" | bc -l)"
 OTHER_SHORT="$(echo $OTHER | sed -r 's/(\..{1}).+/\1/')"
 if [[ "1" == "$(echo "$OTHER < 0" | bc -l)" ]]; then
     OTHER="<span color=\"#FF0000\"><<ledger-main-currency-symbol()>>$OTHER_SHORT</span>"
 else
     OTHER="<span color=\"#00FF00\"><<ledger-main-currency-symbol()>>$OTHER_SHORT</span>"
 fi

 BAL_TIME=$(org-clocked-in.sh balance)
 PROJECTING="0"
 if [[ -f ~/.log/projecting ]]; then
     [[ "$(cat ~/.log/projecting)" == "true" ]] && PROJECTING="1";
 fi
 [[ "$PROJECTING" == "1" ]] && echo "balance.markup=''" | awesome-client || echo "balance.markup='༼ Time:${BAL_TIME} ༼ Food:$BAL_FOOD ༼ Rent:$BAL_RENT ༼ Othr:$OTHER ༼ Fun:$BAL_ENTERTAINMENT_SHORT ༼ Retire:$RETIREMENT ༼ Emergency:$BAL_EMERGENCY ༼ Extra:$EXTRA ༼ ⚖:${INCOME_RATIO}$YEARS_LEFT ${INCOME_RATIO_NO_SALARY}$YEARS_LEFT_NO_SALARY ${INCOME_RATIO_NO_SALARY_NO_GIFTS}$YEARS_LEFT_NO_SALARY_NO_GIFTS'" | awesome-client

The script is ran regularly

while true; do balance-monitor.sh; sleep 100; done &

Ludget - visualisation

pip install ludget

Email client (notmuch)

net-mail/notmuch
net-mail/notmuch doc

Config

:header-args+: :tangle home/yantar92.notmuch-config
[database]
path=/home/yantar92/Mail

[user]
name=Ihor Radchenko
primary_email=yantar92@posteo.net
other_email=ihor_radchenko@alumni.sutd.edu.sg;

[new]
tags=unread;inbox;todo;new;
ignore=

[search]
exclude_tags=deleted

[maildir]
synchronize_flags=true

[header]
List=List-Id

Email composition (emacs)

emacsclient -c -e "(message-mailto \"$@\")"
[Desktop Entry]
Type=Application
Version=1.0
Name=Emacs Client Mailto
Exec=emacs-mailto.sh
Icon=emacs-icon
Terminal=false
MimeType=x-scheme-handler/mailto;

Email sync client (mbsync from net-mail/isync)

Davmail server for Exchange servers (Office365)

[2024-09-03 Tue] Need this for Email in new place.

[2023-01-08 Sun] SUTD disabled my last means to send email using my client. Falling back to redirect + manual WEB interface when necessary. No other options left.

I need to use Microsoft Exchange protocol (everything else was disabled) to access SUTD mail Email from IT Service Desk: RE: IMAP settings for the email server

The best option seems to be Davmail [[id:e8384cfa2557dcfe939e1cf4c3ae80d5b4ae937f][[Reddit] Anyone knows a Mailbox synchronizer supporting Microsoft Exchange protocol? : Gentoo]]

It is not available in Gentoo by default, but I can use pentoo overlay:

eselect repository enable pentoo

Do not enable any other packages from the overlay

*/*::pentoo
net-mail/davmail-bin::pentoo
acct-user/davmail::pentoo
acct-group/davmail::pentoo

Run as server system-wide

net-mail/davmail-bin server
net-mail/davmail-bin
rc-update add davmail default

I will use IMAP Port 10043 in my config. Other important settings: davmail.server=true and davmail.allowRemote=true

[2021-04-14 Wed] Email from IT Service Desk: Re: Case ID: 77322 - Assist on EASE enrolment The new two-factor authentication requires MFA authentication: davmail.mode=O365Modern Details on configuration: [[id:Davmail.Sourceforge_guessantdavmail_pop_imap_smtp_caldav858][Mickael Guessant [Davmail.Sourceforge] DavMail POP/IMAP/SMTP/Caldav/Carddav/LDAP Exchange Gateway - Frequently asked questions]]

davmail.ssl.keystoreType=
davmail.ssl.keystorePass=
davmail.proxyPassword=
davmail.oauth.tenantId=
davmail.oauth.clientId=
davmail.enableKerberos=false
davmail.smtpPort=1025
davmail.folderSizeLimit=
davmail.forceActiveSyncUpdate=false
davmail.imapAutoExpunge=true
davmail.useSystemProxies=false
davmail.proxyUser=
davmail.caldavEditNotifications=false
davmail.ssl.nosecuresmtp=false
davmail.caldavPastDelay=0
davmail.ssl.keyPass=
log4j.logger.httpclient.wire=WARN
davmail.noProxyFor=
log4j.logger.org.apache.commons.httpclient=WARN
davmail.server=true
davmail.popMarkReadOnRetr=false
davmail.ssl.nosecureimap=false
davmail.disableTrayActivitySwitch=false
davmail.caldavAutoSchedule=true
davmail.enableProxy=false
davmail.proxyPort=
davmail.logFileSize=
davmail.mode=O365Modern
davmail.smtpSaveInSent=true
davmail.bindAddress=
davmail.ssl.nosecurepop=false
davmail.ssl.pkcs11Library=
log4j.rootLogger=WARN
davmail.ssl.keystoreFile=
log4j.logger.davmail=DEBUG
davmail.ssl.clientKeystoreType=
davmail.clientSoTimeout=
davmail.ssl.pkcs11Config=
davmail.ssl.clientKeystorePass=
davmail.imapPort=10043
davmail.url=https://outlook.office365.com/EWS/Exchange.asmx
davmail.sentKeepDelay=0
davmail.ssl.nosecureldap=false
davmail.imapAlwaysApproxMsgSize=false
davmail.ssl.nosecurecaldav=false
davmail.popPort=1110
davmail.defaultDomain=
davmail.showStartupBanner=true
davmail.proxyHost=
davmail.ldapPort=1389
log4j.logger.org.apache.http.wire=WARN
davmail.disableGuiNotifications=false
davmail.server.certificate.hash=
davmail.imapIdleDelay=
davmail.allowRemote=true
davmail.disableUpdateCheck=false
log4j.logger.org.apache.http=WARN
davmail.caldavPort=1080
davmail.enableKeepAlive=false
davmail.ssl.clientKeystoreFile=
davmail.logFilePath=/var/log/davmail.log
davmail.carddavReadPhoto=true
davmail.keepDelay=30
davmail.oauth.redirectUri=
davmail.caldavAlarmSound=

Configuration

-----BEGIN PGP MESSAGE-----

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 +pv/d8EP =Ln6v -----END PGP MESSAGE-----

Certificates

Mail checker script

The script checks email from time to time, updates notmuch database and cleans spam. Note that the script does not show notifications about incoming email. This is on purpose. Notifications are terrible for keeping attention on current task. See Digital Minimalism: Choosing a Focused Life in a Noisy World By Cal Newport or [[id:lesswrong_benkuhn2020_tools_keepin_focus][benkuhn [lesswrong] (2020) Tools for Keeping Focused]].

while true; do mail-monitor.sh; sleep 100; done &
while read x; do
    [[ "$x" == "No working address found for" ]] && continue
    [[ "$x" == *"Cannot connect to"* ]] && continue
    [[ "$x" == *"Socket error on"* ]] && continue
    [[ "$x" == *"Temporary failure in name resolution"* ]] && continue
    [[ "$x" == *"Error from IMAP server"* ]] && continue
    [[ "$x" == *"Password is being sent in the clear"* ]] && continue
    [[ "$x" == *"IMAP error: bogus greeting response BAD"* ]] && continue
    [[ "$x" == *"Network is unreachable"* ]] && continue
    [[ "$x" == *"IMAP error: unexpected EOF"* ]] && continue
    [[ "$x" == *"Processed"* ]] && continue
    [[ "$x" == *"pulled"* ]] && continue
    [[ "$x" == *"expunged"* ]] && continue
    [[ "$x" == *"Maildir notice: sleeping due to recent directory modification"* ]] && continue
    [[ "$x" == *"No working address found"* ]] && continue
    [[ "$x" == *"No working address found for 127.0.0.1"* ]] && notify-send "Davmail not working" && continue
    [[ "$x" == *"NO LOGIN failed"* ]] && notify-send "Office365 or Hotmail account blocked. Update password via re-login in browser" && continue
    [[ "$x" == *"Account is blocked. Login to your account via a web browser to verify your identity."* ]] && notify-send "Hotmail account blocked. Login from browser" && continue
    [[ "$x" == *"error"* ]] && notify-send "Unclassified mail checker error: $x" && continue
    [[ "$x" == "" ]] && continue
    [[ "$x" == *"Name or service not known"* ]] && notify-send "$x" && continue
    notify-send "mail-monitor: $x"
    echo $(date): $x >> ~/.log/mail-monitor.log
done
if [[ ! "$1" == "--nofetch" && ! "$PROJECTING" == "true" ]]; then
    mbsync --pull-new -a 2>&1 | mail-monitor-logger.sh;
    # slrnpull -h news2.neva.ru -d ~/Mail/spool/slrnpool/ 2>&1 |  mail-monitor-logger.sh;
fi

notmuch new 1>>/dev/null 2>&1
notmuch-new-messages-list.sh 2>&1 |  mail-monitor-logger.sh

if [[ "$1" == "-v" ]]; then
    # export DBUS_SESSION_BUS_ADDRESS environment variable
    PID="$(pgrep awesome | head -n1)"
    [[ -z "$PID" && -d "/proc/$PID" ]] && export DBUS_SESSION_BUS_ADDRESS="$(grep -z DBUS_SESSION_BUS_ADDRESS /proc/$PID/environ | tr '\0' '\n' | cut -d= -f2-)"
    notify-send "Mail checked"
fi
yant/email-flagged
#if no new messages - exit
MESSAGES="$(notmuch search --output=messages tag:new)"
[[ -z "$MESSAGES" ]] && exit

#echo "`date`"

#delete and ignore in count all "Retrieval using the IMAP4 protocol failed for the following message:"
COUNT_IMAP4="$(notmuch count tag:inbox and \"Retrieval using the IMAP4 protocol failed for the following message:\")"
[[ "$COUNT_IMAP4" == "0" ]] || >&2 echo "Found $COUNT_IMAP4 IMAP4 fail messages"
[[ "$COUNT_IMAP4" == "0" ]] || notmuch tag +deleted -- tag:inbox and "Retrieval using the IMAP4 protocol failed for the following message:"

# clear spam
notmuch-clear-spam.sh

#add tags to the new message in thread
echo "$MESSAGES" | while read x; do
    if [ ! -z "$x" ]; then
	THREAD="`notmuch search --output=threads $x`";
	[[ -z "$THREAD" ]] || TAGS="`notmuch search --output=tags $THREAD`";
	TAGS="`echo $TAGS | sed -e 's/listinbox//g' | sed -e 's/flagged//g' | sed -e 's/draft//g' | sed -e 's/attachment//g' | sed -e 's/signed//g' | sed -e 's/sent//g' | sed -e 's/[0-9]{4}//g' | sed -e 's/forwarded//g'  | sed -e 's/replied//g'`"
	[[ -z "$TAGS" ]] || notmuch tag `echo +\`echo $TAGS | sed -r 's/^[ ]*//'\` | sed -r 's/ / +/g'` +`date | awk '{print $NF}'` -- $x
#	notmuch tag -listinbox -- tag:inbox and $x
    fi
done

#tag usenet group emails
notmuch search --output=messages tag:new | while read message; do
    if [ ! -z "$message" ]; then
	message_file="$(notmuch search --output=files $message)"
	cur_group=$(cat $message_file | grep Newsgroups | cut -d: -f2 | sed -r 's/^ +//');
	if [[ ! -z "$cur_group" ]]; then
	    cur_group="+$(echo $cur_group | sed -r 's/,/ +/g')"
	    # echo "notmuch tag +usenet $cur_group -- $message"
	    notmuch tag +usenet +listinbox -inbox $cur_group -- $message
	    notmuch tag -listinbox +inbox -- $message and tag:nolist
            notmuch tag -inbox -- $message and tag:track
	    THREAD="`notmuch search --output=threads $message`";
	    [[ -z "$THREAD" ]] || THREAD_TAGS="`notmuch search --output=tags tag:deleted and $THREAD`";
	    echo "$THREAD_TAGS" | grep "deleted" && notmuch tag +deleted -- $message
	fi
    fi
done

#tag mailing list emacs
notmuch search --output=messages tag:new | while read message; do
    if [ ! -z "$message" ]; then
	message_file="$(notmuch search --output=files $message)"
	cur_maillist="$(cat $message_file | grep -A1 -i List-Id  | grep '<' | sed -r 's/.*<(.*)>.*/\1/' | head -n1)";
	if [[ ! -z "$cur_maillist" ]]; then
	    # echo "notmuch tag +usenet $cur_maillist -- $message"
	    notmuch tag +maillist +listinbox -inbox +$cur_maillist -- $message
            notmuch tag -listinbox +inbox -- $message and tag:nolist
            notmuch tag -inbox -listinbox -- $message and tag:track
	    THREAD="`notmuch search --output=threads $message`";
	    [[ -z "$THREAD" ]] || THREAD_TAGS="`notmuch search --output=tags tag:deleted and $THREAD`";
	    echo "$THREAD_TAGS" | grep "deleted" >/dev/null && notmuch tag +deleted -- $message
	fi
    fi
done

#Tag flagged
echo "\
<<email-flaglist()>>" | while read x; do
    notmuch tag +inbox +todo +flagged -- $x and tag:new
done

#ARCHIVE some maillists
# ARCHIVE_TAGS="bug-gnu-emacs.gnu.org"
ARCHIVE_TAGS=""

echo "$ARCHIVE_TAGS" | while read x; do
    if [ ! -z "$x" ]; then
	notmuch tag -listinbox -inbox -todo -- tag:$x and tag:new
    fi
done

#Tag messages that must directly go to inbox
notmuch tag +inbox +todo -listinbox -- tag:"/^.+.yantar92.github.com$/" and tag:new
notmuch tag +inbox +todo -listinbox -- tag:emacs-gc-stats.gnu.org and tag:new
notmuch tag +inbox +todo -listinbox -- tag:emacs-orgmode.gnu.org and tag:new and not tag:track
notmuch tag +inbox +todo -listinbox -- tag:~bzg/org-build-failures.lists.sr.ht and tag:new and not tag:track
notmuch tag +inbox +todo -listinbox -- to:yantar92 and tag:new
notmuch tag +inbox +todo -listinbox -- to:Radchenko and tag:new
notmuch tag +inbox +todo -listinbox -- to:Ihor and tag:new

MESSAGES_MAIN="$(notmuch search --output=messages tag:new and not tag:sent and tag:inbox and tag:flagged)"
echo $(notmuch search --output=messages tag:new and not tag:sent and not tag:flagged) | while read message; do
    if [[ ! -z "${MESSAGE}" ]]; then
	THREAD="`notmuch search --output=threads $message`";
	if [[ ! -z "${THREAD}" ]]; then
	    [[ -z "$(notmuch search tag:flagged and $THREAD)" ]] || MESSAGES_MAIN="$MESSAGES_MAIN\n$message";
	fi
    fi
done

#done with initial tagging
notmuch tag -new -- tag:new

[[ -z "$MESSAGES_MAIN" ]] && exit
MESSAGES_NOTIFICATION="$(notmuch search $MESSAGES_MAIN | cut -d' ' -f2- | sed -r 's/^[ ]*//')"
NOTIFICATION="$MESSAGES_NOTIFICATION"
NOTIFICATION=$(echo $NOTIFICATION | sed -E 's/"/\\"/g' | sed -E "s/'/\\\\'/g")
# [[ -z "$MESSAGES_NOTIFICATION" ]] && NOTIFICATION="Usenet:   $COUNT_USENET new messages\nMaillist: $COUNT_MAILLIST new messages" ||\
    # 	NOTIFICATION="Usenet:   $COUNT_USENET new messages\nMaillist: $COUNT_MAILLIST new messages\n"

# PID="$(pgrep awesome | head -n1)"
# [[ -z "$PID" && -d "/proc/$PID" ]] && export DBUS_SESSION_BUS_ADDRESS="$(grep -z DBUS_SESSION_BUS_ADDRESS /proc/$PID/environ | tr '\0' '\n' | cut -d= -f2-)"
touch ~/.log/projecting
[[ "$1" == "--silent" || "$(cat ~/.log/projecting)" == "true" ]] ||  echo "naughty.notify({ title = \"New mail\", text =\"$NOTIFICATION\", timeout = 0, preset = naughty.config.presets.critical})" | awesome-client
yant/email-whitelist
yant/spam-keywords

[2020-12-24 Thu] Disabling. Too many false positives.

Implement reliable debugger for spam filterEND
exit

WHITE_LIST="\
<<email-whitelist()>>"

WHITE_LIST_SEARCH_STRING=$(
    echo "$WHITE_LIST" | while read x; do
	echo " and not from:$x";
    done
			)

#delete and mark as spam all the messages from senders of messages marked as spam
# notmuch address "(tag:deleted AND tag:spam) OR (tag:spam)" | while read x; do
#     EMAIL="$(echo $x | sed -r 's/^.*<(.*)>$/\1/')"; echo "$EMAIL"; done | sort -u | while read x; do
#     [[ -n "$x" ]] && notmuch tag +deleted +spam -- tag:new and not tag:sent and from:$x $WHITE_LIST_SEARCH_STRING;
# done

SPAM_KEYWORDS="\
<<spam-keywords()>>"

echo "$SPAM_KEYWORDS" | while read x; do
    [[ -n "$x" ]] && notmuch tag +deleted +spam -- tag:new and not tag:sent and "$x" $WHITE_LIST_SEARCH_STRING;
done
# echo

#spam servers based on tag
notmuch address "(tag:deleted AND tag:spam_server) OR (tag:spam_server)" | while read x; do
    EMAIL="$(echo $x | sed -r 's/^.*<.*(@.*)>$/\1/')"; echo "$EMAIL"; done | sort -u | while read x; do
    [[ -n "$x" ]] && notmuch tag +deleted +spam -- tag:new and not tag:sent and from:$x $WHITE_LIST_SEARCH_STRING;
done
# echo

# #handle server spam tags
# notmuch search --output messages tag:new and not tag:sent | while read x; do
#     [[ -n "$x" ]] || continue;
#     FILE=$(notmuch search --output files $x)
#     grep "X-Mras: PROBABLE_SPAM" $FILE >/dev/null && notmuch tag +deleted +spam -- $x $WHITE_LIST_SEARCH_STRING
# done

# notmuch search --output files tag:new and not tag:sent | while read x; do
#     ID=$(cat $x | grep "Message-ID" -A1 | tail -n1 | sed -r 's/<//; s/>//; s/[[:space:]]//g')
#     spamassassin < $x | grep "X-Spam-Flag: YES" >/dev/null && notmuch tag +spam +spam_likely -- id:$ID
# done

Mailing list sync (slrn - slrnpull)

net-nntp/slrn

Importing mbox files into maildir

Inspired by https://superuser.com/questions/1169371/how-to-convert-mbox-mail-files-as-found-in-thunderbird-dir-to-maildir Requires notmuch.

import mailbox
import argparse

argparser = argparse.ArgumentParser(
    description="Import mbox file to Maildir",
    epilog="""Author: Ihor Radchenko""",
)

argparser.add_argument(
    "mbox_file", help="mbox file (not archived)")
argparser.add_argument(
    "maildir", help="existing Maildir root")
args = argparser.parse_args()

mbox = mailbox.mbox(args.mbox_file, create=False)
mdir = mailbox.Maildir(args.maildir, create=False)
for x in mbox:
    mdir.add(x)

RSS sync client for websites not providing RSS natively (RSSHub)

Installing using Docker

docker pull diygod/rsshub
docker run -d --name rsshub -p 1200:1200 diygod/rsshub

Generic web-page update watcher (via diff): urlwatch

Installation

www-misc/urlwatch
www-misc/urlwatch ~amd64

[2023-06-16 Fri] Need the same python version for beautifulsoup.

www-misc/urlwatch doc
dev-python/beautifulsoup4 PYTHON_TARGETS: python3_11 python3_10

Configuration

Global config

display:
  error: true
  new: true
  unchanged: false
report:
  email:
    enabled: true
    from: 'yantar92@posteo.net'
    html: true
    method: sendmail
    sendmail:
      path: sendmail
    smtp:
      host: localhost
      keyring: true
      port: 25
      starttls: true
    subject: '{count} changes: {jobs}'
    to: 'yantar92@posteo.net'
  html:
    diff: unified
  pushover:
    app: ''
    enabled: false
    user: ''
  stdout:
    color: true
    enabled: true
  text:
    details: true
    footer: true
    line_length: 75
job_defaults:
  url:
    ignore_connection_errors: true

URLs to watch

-----BEGIN PGP MESSAGE-----

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 EzkuN8xpJMv8sN4= =UmQI -----END PGP MESSAGE-----

Mail sending (msmtp)

mail-mta/msmtp

Config

-----BEGIN PGP MESSAGE-----

hQEMA3SKBiQ2zhL6AQf/SuYAM/nekBnyd11PHpZFLqL/uEf1bZjLTORRwjsHAA50 LwReYfSt5QHWTc1Cw7tqXyli2p3ZrTx8Q5IQgbVxWiY+Q0+5JrtYR1i2AyCvrjb3 fp8j6umnDwW5QHPUVE61WcCQhO15ZY5Am2WrrvBZLojAtC51yLdsc6tADSVgnY8q QNK9HShB0zj1P3csD0n5vdEZJYrXBxv3+snqaONB7TePsRsCn9TMT3TpBDxYoRcq i/F9DXp/1EaTNCBA77rXcVOZq8yyBh5MsN/QlTJAfy6iwiNoj2KmhJf20opOMxPu YQm4JXyNDn3bqsyJDjIKnnOlBW3Rvrshm2mlT9nBcdLpAUycYJKSCazilkTDRkVJ qlU67L2AZ5UcfMrcOcgratNmW0OCrV6WgPqDZyTyOEnuclEz8DtngNvobWHN+4SJ efO4wSE7DR3YxVZWgdmzj9tiHxykgN1wLcVxUher2QBSPiT1sqNLu8T17qf5Ok14 Kds4xmCRRN6AjOjjf0JZZ+LTpn9IzIIpyz/Ub/O7k4i1JhS/m63XQmBa8PWuN+tU aBIBxodSR5NbgL6UvGHrz6dafcYT4fHglGFhtksHxLV7nZ7GNEozka8jGILLxNId KJ9kesEa8pkMy0z8Lf3qKwVCvVkpfhkvQBILkPmn3ec2THGhMyqF7n/tJX0B+CKb Vm54DoWo8QavWpVUMgaX7uFu4OEctz2xtNSy2l4ifCFU9MadRhOtiK1W8lJj7bxo SftOjCTUmTJHw4K41Z5ZmMO9mMirnetftJw7c1KLYFH7oS+XY9DM2AFUFKG5v1Xs tV68y6S/e6x7Z3vM9eMRnFb2ultkpPvcT5+oVif9gFE7icoO4KhT06Fh2U+sx7uO tap2YSs3zE5zs6AfWXV4MH8ttAuxBLigveNkMeYLtykda0cnNYMq9XkRSe5Fo1IF xHvf01yYrkFVoAlDhuHm9VTdJvqzUvIP5CcvVlxPUb7daIMJACiM4EM1a1f26f/j H72PLg6+CEnenoDotK3h73XAIkQqZjBWynKoNTcrbv5RM3eLcoPqj2/6VZjU9U8z og6PyGcJdLnSL4Uk0wx9oCMLQHkCwdfJWqnWiEuJE/Rc26MoVALrOUs8OCGwKX+z TOQOOZMVZ9nOmW8OwthDN4QzD+FeXfTebhJ70974RbszjISVNnOqZbtEAIGPoZ9P f5cboAL4YorIaGBxEVg0O8QUzXb6ih7CSFkvVQoeWhFGg3Vx2OVpjvyWWB2AA2hU aMBwvW3A0XsrcrzdhsCZ6JnU2bRnKrOXK1RDIjHTxqwhGo1ouuLdXZ1CvTu1ched MrYbtY8= =3Jty -----END PGP MESSAGE-----

Dictionary: translate-shell

app-i18n/translate-shell

Show dictionary in rofi

word="$(rofi -lines 0 -p 'dictionary: ' -dmenu)"
if [[ ! -z "$word" ]]; then
    rofi -width 800 -location 2 -e "$(trans -w 400 --no-ansi :en \\"$word\\")"
fi

Show translation in rofi

word="$(rofi -lines 0 -p translate: -dmenu)"
if [[ ! -z "$word" ]]; then
    rofi -width 800 -location 2 -e "$(trans -w 400 --no-ansi :ru $word)"
fi

Password storage: pass

app-admin/pass

Rofi integration

I have a custom script allowing to get passwords (without copying to clipboard) Requires Xdotool and setxkbmap

shopt -s nullglob globstar

typeit=0
user=0
while (( $# > 0 )); do
    if [[ $1 == "--type" ]]; then
	       typeit=1
	       shift
	   fi
	   if [[ $1 == "--user" ]]; then
	       user=1
               shift
	   fi	
done

prefix=${PASSWORD_STORE_DIR-~/.password-store}
password_files=( "$prefix"/**/*.gpg )
password_files=( "${password_files[@]#"$prefix"/}" )
password_files=( "${password_files[@]%.gpg}" )

password=$(printf '%s\n' "${password_files[@]}" | rofi -dmenu -i "$@")

[[ -n $password ]] || exit

if [[ $user -eq 0 ]]; then
    if [[ $typeit -eq 0 ]]; then
	pass show -c "$password" 2>/dev/null
    else
	pass show "$password" | { read -r pass; printf %s "$pass"; } |
	    (setxkbmap us; xdotool type --clearmodifiers --file -; setxkbmap us,ua)
    fi
else
    USERNAME=$(pass show "$password" | grep -i user | cut -d: -f2 | sed 's/^[ ]*//')
    if [[ $typeit -eq 0 ]]; then
	echo "$USERNAME" | xclip 
    else
	echo "$USERNAME" | { read -r username; printf %s "$username"; } |
	    (setxkbmap us; xdotool type --clearmodifiers --file -; setxkbmap us,ua)
    fi    
fi

Anki: spaced repetition software

app-misc/anki
app-misc/anki ~amd64

[2021-04-04 Sun] Adding dbus use-flag for qutebrowser

dev-python/PyQt5 network webchannel printsupport declarative multimedia sql dbus

The card database is stored in file:~/.local/share/Anki2

Also, need to install dev-python/pyaudio to play sounds.

dev-python/pyaudio

Gnuplot

cairo is needed for [[id:438648e38c40bbdff091d416e59c243d44da93a5][yantar92 [Github] video-graph: A simple script to combine load/displacement data with video recording from Hysitron PI picoindenter]]

sci-visualization/gnuplot
sci-visualization/gnuplot doc examples qt5 cairo

Configuration

Based on other templates.

Create reference from DistEND
# palette
set palette defined (0 "violet",1 "blue",2 "cyan",3 "green",4 "yellow",5 "orange",6 "red")

# Init Variables
GRIDCOLOR = "#000000"
TEXTCOLOR = "#000000"
LINEWIDTH = 3
POINTSIZE = 0.5
# if (!exists('LOCALE')) {
#      LOCALE = system('echo $LANG')
# }

#set locale LOCALE
#set decimalsign locale LOCALE

POINTTYPE1 = 7
POINTTYPE2 = 5
POINTTYPE3 = 9
POINTTYPE4 = 13
POINTTYPE5 = 11
POINTTYPE6 = 7
POINTTYPE7 = 2
POINTSIZE1 = 1.0*POINTSIZE
POINTSIZE2 = 0.9*POINTSIZE
POINTSIZE3 = 1.2*POINTSIZE
POINTSIZE4 = 1.1*POINTSIZE
POINTSIZE5 = 1.2*POINTSIZE
POINTSIZE6 = 1.0*POINTSIZE
POINTSIZE7 = 0.9*POINTSIZE

set linetype 1 linecolor rgb "#9932CC" linewidth LINEWIDTH pointtype POINTTYPE1 pointsize POINTSIZE1
set linetype 2 linecolor rgb "#1F78B4" linewidth LINEWIDTH pointtype POINTTYPE2 pointsize POINTSIZE2
set linetype 3 linecolor rgb "#B2DF8A" linewidth LINEWIDTH pointtype POINTTYPE3 pointsize POINTSIZE3
set linetype 4 linecolor rgb "#33A02C" linewidth LINEWIDTH pointtype POINTTYPE4 pointsize POINTSIZE4
set linetype 5 linecolor rgb "#FB9A99" linewidth LINEWIDTH pointtype POINTTYPE5 pointsize POINTSIZE5
set linetype 6 linecolor rgb "#773355" linewidth LINEWIDTH dashtype 2 pointtype POINTTYPE7 pointsize POINTSIZE7
set linetype 7 linecolor rgb "#000000" linewidth LINEWIDTH dashtype 1 pointtype POINTTYPE6 pointsize POINTSIZE6

set linetype cycle 7

DATA="LINEWIDTH=0.5*POINTSIZE"
FUNC="LINEWIDTH=3"
@DATA

PNG_ORG="set term pngcairo enhanced font ',12' enhanced transparent size 1280,960 fontscale 1.7 lw 2.3 dashlength 0.5; GRIDCOLOR = \"#ffffff\"; TEXTCOLOR = \"#ffffff\"; load \"/home/yantar92/.gnuplot\""
PNG="set term pngcairo enhanced font ',12' enhanced size 1280,960 fontscale 1.7 lw 2.3 dashlength 0.5"
QT="set term qt font ',10' size 1280,960 fontscale 1.7 lw 2.3 dashlength 0.5"

unset xtics
unset ytics
unset mxtics
unset mytics
unset key
unset grid
unset border
unset xlabel
unset ylabel
unset label 100
unset label 101
unset label 200
unset label 201
unset arrow 100
unset arrow 200

unset lmargin
unset bmargin
unset tmargin
unset rmargin

set xrange [*:*]
set yrange [*:*]
set zrange [*:*]

# Tics
set xtics nomirror in scale 1,0.5 offset 0,0 autofreq textcolor rgb TEXTCOLOR
set ytics mirror in scale 1,0.5 offset 0,0 autofreq textcolor rgb TEXTCOLOR
set tics font ",12"

# Key
set key inside left top Left revers samplen 2 spacing 1.3 textcolor rgb TEXTCOLOR

# Grid
set grid noxtics ytics lt 0 lw 1 linecolor rgb GRIDCOLOR
set mxtics 5
set mytics 5

# Border
set border 11 lt -1 lw 2 linecolor rgb GRIDCOLOR

# Labels
#set label 200 '200: ylabel' at screen 0, graph 0.5 center offset 1,0 textcolor rgb TEXTCOLOR rotate by 90
#set label 201 '' at screen 1, graph 0.5 center offset -1,0 textcolor rgb TEXTCOLOR rotate by 90
#set label 100 '100: xlabel' at graph 0.5, graph 0 center offset 0,-3 textcolor rgb TEXTCOLOR
#set label 101 '' at graph 0.5, graph 1 center offset 0,3 textcolor rgb TEXTCOLOR

#Margin
set lmargin 10
set bmargin 4


# Ranges
set xrange [*<10:*]
set yrange [*<10:*]
set zrange [*<10:*]

#axes labels font size
set xlabel offset 0,0.0 textcolor rgb TEXTCOLOR font ",14"
set ylabel offset 0,0.0 textcolor rgb TEXTCOLOR font ",14"
set x2label offset 0,0.0 textcolor rgb TEXTCOLOR font ",14"
set y2label offset 0,0.0 textcolor rgb TEXTCOLOR font ",14"
set zlabel offset 0,0.0 textcolor rgb TEXTCOLOR font ",14"
set cblabel textcolor rgb TEXTCOLOR font ",14" offset char 1, 0


# for Latex Terminals load the latex settings
if (strstrt(GPVAL_TERM, 'latex') > 0) {
# Settings for Latex Terminals

set format "\\num{%g}" # Set number fomrat to siunitx

# Lables

arrowLabel(str) = sprintf("%s {\\tikz[baseline] \\draw[-latex,thick] (0,0.5ex) -- (7ex,0.5ex);}",str)

shortLabel(symb,unit) = sprintf("$%s$ in \\si{%s}",symb,unit)
shortArrowLabel(symb,unit) = arrowLabel(shortLabel(symb,unit))

longLabel(desc,symb,unit) = sprintf("%s %s",desc,shortLabel(symb,unit))
longArrowLabel(desc,symb,unit) = arrowLabel(longLabel(desc,symb,unit))

shortPuLabel(symb,unit) = sprintf("$\\nicefrac{%s}{%s}$",symb,unit)
shortPuArrowLabel(symb,unit) = arrowLabel(shortPuLabel(symb,unit))

longPuLabel(desc,symb,unit) = sprintf("%s %s",desc,shortPuLabel(symb,unit))
longPuArrowLabel(desc,symb,unit) = arrowLabel(longPuLabel(desc,symb,unit))

shortLabel(symb,unit) = sprintf("$\\nicefrac{%s}{\\si{%s}}$",symb,unit)

}

Presentation: pdfpc

write custom ebuildEND Need gstreamer support for video. Also, need to install additional packages: gst-plugins-meta and media-plugins/gst-plugins-gtk
app-text/pdfpc
media-plugins/gst-plugins-meta
media-plugins/gst-plugins-gtk

The video support for gstreamer can be tested using the following command [[id:644ac652e7528eb6af7fb03e3ecb168639846dbb][olivier-klein [Github] issue#547 pdfpc and Gstreamer]]:

gst-play-1.0 --videosink gtksink apollo17.avi

The result should be playing video and no warnings about =gtksink= creation.

Some of the below flags are probably excessiveEND
app-text/pdfpc gstreamer
media-plugins/gst-plugins-base theora
media-plugins/gst-plugins-meta ffmpeg theora ogg dvb
app-text/pdfpc ~amd64

Python lib for automatic language detection: langdetect

Need dev-python/pip.

pip install langdetect

Syncthing

net-p2p/syncthing

Startup on load. Configuration is copied from old laptop.

syncthing -no-browser 2>&1 | syncthing-logger.sh &
while read x; do
    [[ "$x" == *"INFO:"* ]] && continue
    [[ "$x" == "" ]] && continue
    echo $(date): $x >> ~/.log/mail-monitor.log
done

Drawing

Gimp

media-gfx/gimp

Inkscape

media-gfx/inkscape

[2020-12-13 Sun] Inkscape does not support tiff images by default. Adding imagemagick use-flag.

# Required for installation
dev-cpp/gtkmm X
dev-cpp/cairomm X
media-gfx/inkscape imagemagick postscript jpeg inkjar svg2

Git

Global attributes file

git config --global core.attributesfile ~/.gitattributes
*.lisp  diff=lisp
*.el    diff=lisp
*.org   diff=org

Main config

[2020-11-08 Sun] Added diff handler to name diff chunks by headline in org files [2021-04-01 Thu] Change regexp according to [[id:7512681c13307f3005696422fa21170a171f538f][Protesilaos Stavrou: Coding blog [Protesilaos] (2021) Informative diff hunks for Emacs Lisp and Org]]; add lisp regexp

[user]
	email = yantar92@posteo.net
	name = Ihor Radchenko
        signingkey = 6470762A7DA11D8B
[github]
	user = yantar92

[commit]
  gpgsign = true
  
[merge]
  conflictstyle = diff3

[format]
  thread = true

[diff "lisp"]
  xfuncname = "^(((;;;+ )|\\(|([ \t]+\\(((cl-|el-patch-)?def(un|var|macro|method|custom)|gb/))).*)$"

[diff "org"]
  xfuncname = "^(\\*+ +.*)$"
    
[core]
	attributesfile = /home/yantar92/.gitattributes

Rebase by default for new tracking branches

When doing feature development, it makes little sense to merge the feature branch to master explicitly using separate commit. It is better to use rebase instead of merge commit, as recommended in [[id:Stack_Overflow_autos_vs_autosb9a][[Stack Overflow] autosetuprebase vs autosetupmerge]].

[branch]
  autosetupmerge = always
  autosetuprebase = always

TeX \ LaTeX (texlive)

app-text/texlive

Use XeTeX and use many optional packages. They are not needed most of the time, but I remember myself regretting that I do not have these extra installed several times - they take a long time to install even if needed urgently.

app-text/texlive xetex games humanities publishers science pstricks png l10n_ru l10n_uk extra luatex graphics
app-text/texlive-core doc xetex
dev-texlive/texlive-publishers doc
dev-texlive/texlive-games doc 
dev-texlive/texlive-humanities doc
dev-texlive/texlive-langcjk doc
dev-texlive/texlive-langenglish doc

dev-texlive/texlive-fontsextra

dev-texlive/texlive-langenglish: Info documentation (docs) for LaTeX (inspired by [[id:ML:Org-mode-<theophilusx@gmail.com>2021-re-quick-latex-77d][Tim Cross [ML:Org mode] (2021) Re: A quick LaTeX reference guide in Org]]).

Important: I had to manually install the relevant info file:

cd usr/share/texmf-dist/doc/info
install-info ./latex2e.info ./dir

latexmk: easier building of TeX documents

dev-tex/latexmk

Packages

biblatex: better citations

dev-tex/biblatex

#documentclass memoir: simple formatting

#margins #font_size #documentclass

Manual: memoir manual

latexdiff: View changes in generated pdf

dev-tex/latexdiff
dev-tex/latexdiff **

Latex preview for Org dvisvgm

app-text/dvisvgm

Image viewer: feh

media-gfx/feh

Call wrapper:

DIRNAME="`dirname "$1"`"
if [[ -z "$DIRNAME" ]]; then DIRNAME="$PWD"; fi
FILENAME="`dirname "$1"`/`basename "$1"`"

# FILES="$(find "$DIRNAME/" -maxdepth 1 | sort)"
# FILENUMBER="$(echo "$FILES" | awk '{print NR" "$0}' | grep "$FILENAME" | cut -d' ' -f1)"
# FILENUMBER=$(echo "$FILENUMBER - 1" | bc -l)
# sxiv -n $FILENUMBER "$DIRNAME/"*
feh -d -B white --scale-down --auto-zoom --draw-exif --draw-tinted --edit --version-sort --start-at  "$FILENAME" "$DIRNAME" 2>&1 > /dev/null

Desktop file

[Desktop Entry]
Version=1.0
Type=Application
Name=feh-open
Exec=feh-open %f
TryExec=feh
Terminal=false
StartupNotify=true
MimeType=image/bmp;image/gif;image/x-tga;image/x-xbitmap;image/tiff;image/jpeg;image/x-psp;image/png;image/x-icon;image/webp;

Video

mpv

media-video/mpv

Had to copy over the saved information from ~/.config/mpv/watch_later

media-video/mpv cli

Config

:header-args+: :tangle no
Main config
:header-args+: :tangle home/yantar92.config/mpv/mpv.conf :mkdirp yes
General

Open window without waiting for video to load (relevant when opening online videos on high-ping connection). Do not auto-play for the same reason - sometimes I need to wait for a quite a long time and suddenly playing video would be a surprise.

force-window=immediate
pause                                   # no autoplay

Save position

save-position-on-quit

Initial window placement: center

geometry=1440x900+50%+50%

Save screenshots in PNG format into Downloads folder for further refiling.

screenshot-format=png
screenshot-template='~/Downloads/%F (%P) %n'
keepaspect-window=no
loop-playlist=yes
msg-module                              # prepend module name to log messages
msg-color                               # color log messages on terminal
term-osd-bar                            # display a progress bar on the terminal
keep-open                               # keep the player open when a file's end is reached
Cache

I prefer huge cache to preload online videos if possible. Connection is not always stable.

cache=yes
cache-secs=1800                           # how many seconds of audio/video to prefetch if the cache is active
prefetch-playlist=yes
Subtitles

Automatically detect subtitle file

sub-use-margins
sub-auto=fuzzy                          # external subs don't have to match the file name exactly to autoload
sub-file-paths=ass:srt:sub:subs:subtitles    # search for external subs in the listed subdirectories
embeddedfonts=yes                       # use embedded fonts for SSA/ASS subs
sub-fix-timing=no                       # do not try to fix gaps (which might make it worse in some cases)

# the following options only apply to subtitles without own styling (i.e. not ASS but e.g. SRT)
sub-font="Helvetica"
sub-font-size=36
sub-color="#FFFFFFFF"
sub-border-color="#FF262626"
sub-border-size=3.2
sub-shadow-offset=1
sub-shadow-color="#33000000"
sub-spacing=0.5
Languages
slang=enm,en,eng,ru             # automatically select these subtitles (decreasing priority)
alang=enm,en,ang,ru       # automatically select these audio tracks (decreasing priority)
Audio
af=scaletempo=speed=tempo:stride=30
audio-file-auto=fuzzy                   # external audio doesn't have to match the file name exactly to autoload
Protocols

yt-dlp

script-opts=ytdl_hook-ytdl_path=/usr/bin/yt-dlp
ytdl-format=bestvideo[height<=720]+bestaudio
[protocol.https]
user-agent='Mozilla/5.0 (Macintosh; Intel Mac OS X 10.11; rv:47.0) Gecko/20100101 Firefox/47.0'

[protocol.http]
user-agent='Mozilla/5.0 (Macintosh; Intel Mac OS X 10.11; rv:47.0) Gecko/20100101 Firefox/47.0'

[extension.gif]
cache=no
no-pause
loop-file=yes
Key bindings
:header-args+: :tangle home/yantar92.config/mpv/input.conf
ctrl+q quit
q ignore
User-scripts
YouTube
Change quality

C-f to call menu.

https://github.com/jgreco/mpv-youtube-quality

cat "$file"
<<get-file()>>
# KEY BINDINGS

# invoke or dismiss the quality menu
toggle_menu_binding=ctrl+f
# move the menu cursor up
up_binding=Alt+k
# move the menu cursor down
down_binding=Alt+j
# select menu entry
select_binding=Ctrl+j

# formatting / cursors
selected_and_active=▶ - 
selected_and_inactive=● - 
unselected_and_active=▷ - 
unselected_and_inactive=○ - 

# font size scales by window, if false requires larger font and padding sizes
scale_playlist_by_window=yes

# playlist ass style overrides inside curly brackets, \keyvalue is one field, extra \ for escape in lua
# example {\\fnUbuntu\\fs10\\b0\\bord1} equals: font=Ubuntu, size=10, bold=no, border=1
# read http://docs.aegisub.org/3.2/ASS_Tags/ for reference of tags
# undeclared tags will use default osd settings
# these styles will be used for the whole playlist. More specific styling will need to be hacked in
#
# (a monospaced font is recommended but not required)
style_ass_tags={\\fnmonospace}

# paddings for top left corner
text_padding_x=5
text_padding_y=5

# how many seconds until the quality menu times out
menu_timeout=10

#use youtube-dl to fetch a list of available formats (overrides quality_strings)
fetch_formats=yes

# list of ytdl-format strings to choose from
quality_strings=[ {"4320p" : "bestvideo[height<=?4320p]+bestaudio/best"}, {"2160p" : "bestvideo[height<=?2160]+bestaudio/best"}, {"1440p" : "bestvideo[height<=?1440]+bestaudio/best"}, {"1080p" : "bestvideo[height<=?1080]+bestaudio/best"}, {"720p" : "bestvideo[height<=?720]+bestaudio/best"}, {"480p" : "bestvideo[height<=?480]+bestaudio/best"}, {"360p" : "bestvideo[height<=?360]+bestaudio/best"}, {"240p" : "bestvideo[height<=?240]+bestaudio/best"}, {"144p" : "bestvideo[height<=?144]+bestaudio/best"} ]

#+RESULTS[33e778796fc04ae9b0a18b218e68bf5c5d3a404d]:

Annie - better than youtube-dl for bilibili

[[id:3ae439f24f6fba59189a9b254cab318c6fb26352][muzea [Github] annie: 👾 Fast, simple and clean video downloader]]

yt-dlp: maintained fork of youtube-dl

net-misc/yt-dlp

Wine

  • install samba with winbind useflag - needed for some programs
virtual/wine
app-emulation/dxvk
app-emulation/vkd3d-proton
setup_dxvk.sh install --symlink
setup_vkd3d_proton.sh install --symlink
net-fs/samba winbind
virtual/wine proton

Scientific image manipulation: Fiji \ [[id:5aa36a1d631ea6266284bbc74d3591217237a36a][[Imagej] ImageJ]]

Zoom (video conference) net-im/zoom

[[id:34c0096a61b89a7a56c1b8670552140048d01367][[Zoom] Video Conferencing, Web Conferencing, Webinars, Screen Sharing - Zoom]]

I don’t like it, but it is what I have to use

net-im/zoom all-rights-reserved
net-im/zoom

GNU Screen (for Org mode ob-screen testing)

app-misc/screen

Music editing

Split flac file according to cue playlist

Using media-sound/shntool

media-sound/shntool
media-sound/shntool alac flac wavpack
[[ $# == 2 ]] && shnsplit -f "$1" -o flac -t "%n-%t" "$2" || echo "Wrong number of arguments: specify cue file and flac file"

Video editing

Convert video to format suitable for LibreOffice impress

help() {
    echo "\
Convert video to the ogg vorbis format acceptable to insert into libreoffice imrpess presentations
Usage: $0 [-h] [-vq NUM] [-aq NUM] <input file> [output file.ogv]
-h|--help: show help
-vq <0-10>: video quality, default - 7
-au <0-10>: audio quality, default - 5"
    }
VQ="7";
AQ="5";
INPUT="";
OUTPUT=""
PARSE=0
while [[ "$PARSE" == "0" ]]; do
    case "$1" in
	-h|--help)
	    help
	    exit 1
	    ;;
	-vq)
	    shift
	    if [[ $! =~ '^[0-9]+$' ]]; then
		VQ="$1";
	    else
		echo "Not a number: $1";
		exit 1;
	    fi
	    shift
	    ;;
	-aq)
	    shift;
	    if [[ $! =~ '^[0-9]+$' ]]; then
		AQ="$1";
	    else
		echo "Not a number: $1";
		exit 1;
	    fi
	    shift
	    ;;
	*)
	    PARSE=1;
	    break
	    ;;
    esac
done
INPUT="$1";
if [[ ! -f "$INPUT" ]]; then
    echo "File does not exist: \"$INPUT\"";
    exit 1;
fi
shift;
(( $# > 0 )) && OUTPUT="$1" && shift
OUTPUT=${OUTPUT:-${INPUT%.*}.ogv}
[[ ! "${OUTPUT##*.}" == "ogv" ]] && echo "Output file \"$OUTPUT\" should have ogv extension" && exit 1
(( $# > 0 )) && echo "Too many arguments: $*" && exit 1
ffmpeg -hide_banner -i "$INPUT" -codec:v libtheora -qscale:v $VQ -codec:a libvorbis -qscale:a $AQ "$OUTPUT"

Speed up video

help="Change video speed
Usage: $0 [-h|--help] [-an] [-o output] [-fps FPS] input_file speed_multiplier
-an: remove audio
-o: specify output (default: [speed_multiplierx]input_file)
-fps: specify fps (default: preserve all the frames)
-h|--help: show help"
[[ $# < 2 ]] && echo "$help" && exit
while [ "$#" -gt "2" ]; do
    case "$1" in
	"-an")
	    OPT="-an"
	    shift
	    ;;
	"-o")
	    output_file="$2"
	    shift
	    shift
	    ;;
	"-fps")
	    FPS="$2"
	    [[ ! "$FPS" -eq "$FPS" ]] && echo "Not a number: $FPS" && echo "$help" && exit
	    shift
	    shift
	    ;;
	"-h"|"--help")
	    echo "$help"
	    exit
	    ;;
	*)
	    echo "Unknow argument: #1"
	    echo "$help"
	    exit
	    ;;
    esac
done
input_file="$1"
mult="$2"
[[ -z "$output_file" ]] && output_file="[${mult}x]`basename \"$input_file\"`"
[[ -z "$FPS" ]] && FPS="`ffprobe \"$input_file\" 2>&1 | grep -oe \"[0-9]* fps\" | grep -oe \"[0-9]*\"`" && FPS="`echo \"$FPS * $mult\" | bc -l`"
INV="`echo \"1 / $mult\" | bc -l`"
if [[ "$OPT" == "-an" ]]; then
    ffmpeg -hide_banner -i "$input_file" $OPT -r $FPS -filter_complex "[0:v]setpts=$INV*PTS" "$output_file"
else
    ffmpeg -hide_banner -i "$input_file" -r $FPS -filter_complex "[0:v]setpts=$INV*PTS; [0:a]asetpts=$INV*PTS" "$output_file"
fi

Convert video to mkv without lossy compression

I need this for research videos where lossy compression is not acceptable. Assumes that mkv format is supported.

media-video/ffmpeg x264
[[ $# = 2 ]] && ffmpeg -v error -hide_banner -i "$1" -f lavfi -i "anullsrc=cl=1" -shortest -crf 0 -filter_complex  "[0:v]scale=trunc(iw/2)*2:trunc(ih/2)*2" -c:v libx264 "$2" || echo "Wrong number of arguments. Usage: video2mkv.sh input_file output_file"

Convert video to raw avi format without lossy compression

function help() {
    echo "Usage: video2rawavi.sh input_file [output_file]" 
}
if [[ $# < 1 || $#>2 ]]; then
    echo "Wrong number of arguments: $#";
    help
    exit
fi
output=""
[[ $# = 1 ]] && output="$(echo $1 | sed -r 's/\.[^.]+$/\.avi/')"
if [[ ! -f "$1" ]]; then
    echo "File \"$1\" does not exists";
    exit;
fi
[[ $# = 2 ]] && output="$2";
[[ -z "$output" ]] || ffmpeg -v error -hide_banner -i "$1" -c:v rawvideo -pix_fmt bgr24 "$output"

Concat several videos into one

INPUT=""
OUTPUT=""
while [[ $# > 1 ]]; do
    [[ -z "$INPUT" ]] && INPUT="file '$1'" || INPUT="${INPUT}
file '${1}'"
    shift
done
OUTPUT="$1"
echo -e "$INPUT" > list.txt
ffmpeg -f concat -safe 0 -i list.txt -c copy "$OUTPUT"
rm list.txt

Extract segment from a video

#!/bin/bash
function help() {
    echo "$(basename $0): Clip video
    Usage: $0 [-o output] [--from start_position] file duration
    Defaults: start_position=0, output=[clipped]file"
}
[[ $# < 2 ]] && echo "Too less arguments" && help && exit
while [[ $# > 0 ]]; do
    case "$1" in
	"-h"|"--help")
	    help
	    exit
	    ;;
        "-o")
	    output="$2"
	    shift
	    ;;
        "--from")
	    start="$2"
	    shift
	    ;;
        *)
	    file="$1"
	    if [[ ! -f "$file" ]]; then
		echo "File not found \"$file\""
                exit
	    fi		
	    duration="$2"
	    [[ -z "$output" ]] && output="$(dirname "$file")/[clipped]$(basename "$file")"
	    shift
	    ;;
    esac
    shift
done
ffmpeg $([[ -z "$start" ]] || echo "-ss $start") -i "$file" -c copy -t $duration "$output"

Extend video duration to some target

This is useful helper script when I need to combine several videos side-by-side.

function help() {
    echo "$(basename $0): extend video duration by repeating the last frame
    Usage: $0  [-o video_output] video_file new_time
    new_time: [hours]:[minutes]:seconds
    -o: specify output file name (default: [new_time]video_file)"
}

hours=0
minutes=0
seconds=0.1

function parse_duration () {
    regexp_full="(([0-9]+):)(([0-9]+):)([0-9]+(\.[0-9]+)?)"
    regexp_nohours="(([0-9]+):)([0-9]+(\.[0-9]+)?)"
    regexp_onlyminutes="([0-9]+(\.[0-9]+)?)"
    full="$(echo "$1" | grep -E "$regexp_full")"
    nohours="$(echo "$1" | grep -E "$regexp_nohours")"
    onlyminutes="$(echo "$1" | grep -E "$regexp_onlyminutes")"
    if [[ ! -z "$full" ]]; then
	parsed_duration="$(echo "$1" | sed -r "s/$regexp_full/\2 \4 \5/")"
    elif [[ ! -z "$nohours" ]]; then
		parsed_duration="$(echo "$1" | sed -r "s/$regexp_nohours/0 \2 \3/")"
    elif [[ ! -z "$onlyminutes" ]]; then
	parsed_duration="$(echo "$1" | sed -r "s/$regexp_onlyminutes/0 0 \1/")"
    else
        echo "Wrong duration: \"$1\"" && exit
    fi
    hours=$(echo "$parsed_duration" | awk '{print $1}')
    minutes=$(echo "$parsed_duration" | awk '{print $2}')
    seconds=$(echo "$parsed_duration" | awk '{print $3}')
}

[[ $# < 2 ]] && help && exit
output_file=""
while [[ $# > 0 ]]; do
    case "$1" in
	"-o")
	    output_file="$2"
	    shift
	    shift
	    ;;
        "-h"|"--help")
	    help
	    exit
	    ;;
	*)
	    input_file="$1"
	    new_duration="$2"
	    [[ -z "$output_file" ]] && output_file="$(dirname $input_file)/[$new_duration]$(basename $input_file)"
	    shift
	    shift
	    ;;
    esac
done
if [[ ! -f "$input_file" ]]; then
    echo "File not found: \"$input_file\"";
    exit
fi

duration=$(ffprobe "$input_file" 2>&1 | grep -Eo "Duration: [^,]+" | sed -r 's/Duration: //')
parse_duration "$duration"
duration_sec=$(echo "${hours}*60*60+${minutes}*60+$seconds" | bc -l)
parse_duration "$new_duration"
new_duration_sec=$(echo "${hours}*60*60+${minutes}*60+$seconds" | bc -l)
[[ "$(echo "$new_duration_sec < $duration_sec")" == "1" ]] && echo "New duration (\"$new_duration\") is smaller than video duration (\"$duration\")" && exit

fps="$(ffprobe $input_file 2>&1 | grep -Eo "[0-9]+ fps" | cut -d' ' -f1)"
last_frame="$(echo "${fps}*${duration_sec} - 1" | bc -l | sed -r 's/\..+//')"


ffmpeg -i "$input_file"  -vf "select='eq(n,$last_frame)'" -vframes 1 /tmp/last_frame.png &&\
    ffmpeg -loop 1 -i /tmp/last_frame.png -c:v libx264 -r $fps -t $(echo "$new_duration_sec - $duration_sec" | bc -l | sed -r 's/^\./0\./') -pix_fmt yuv420p /tmp/last_frame.mp4 &&\
    video_concat.sh "$input_file" /tmp/last_frame.mp4 "$output_file" && rm /tmp/last_frame.png /tmp/last_frame.mp4

Conversion

djvu to pdf

app-text/djvu
INPUT="$1"
if [[ ! -f "$INPUT" ]]; then
   echo "File not exist \"$INPUT\""
   exit 1
fi
OUTPUT="$2"
if [[ -f "$2" ]]; then
    echo "File exists \"$2\""
    exit 1
fi
OUTPUT=${OUTPUT:-${INPUT%.*}.pdf}
djvups "$INPUT" "/tmp/$(basename \"$INPUT\").ps" && ps2pdf "/tmp/$(basename \"$INPUT\").ps" "$OUTPUT" && rm "/tmp/$(basename \"$INPUT\").ps" || (echo "Something wrong"; exit 1)

Presentation setup: silence unnecessary notifications

It is extremely bad idea to show distracting notifications during presentation. The script below tries to connect projector, stop all composition (to avoid any possible glitches), stops conky, and notifies all relevant scripts that projecting is in progress.

Also, I need to install x11-apps/xrandr to control plugged projector.

x11-apps/xrandr
xrandr --output HDMI-1 --auto --right-of eDP-1
pgrep conky && conky-stop
echo "true" > ~/.log/projecting
balance-monitor.sh
xrandr --output HDMI-1 --off
# conky-start
echo "false" > ~/.log/projecting
balance-monitor.sh

Backup

[2020-12-25 Fri] Switching to restic

Exclude uninteresting dirs and also notmuch dir.

(concat yant/backup-script-opts " --exclude /home/yantar92/Mail/.notmuch")

Notmuch tags are backed up separately, using plain text file rather than xapian binary db. Binary db is not best for incremental backups adding 1-2Gb unnecessarily.

cd /home/yantar92/Mail/
mv notmuch-dump.gz notmuch-dump.gz.bak
sudo -u yantar92 notmuch dump --gzip --output=notmuch-dump.gz
# Store to incremental backup
while true; do
    read -p "Sync with local backup? " yn
    case $yn in
        [Yy]* ) restic -r /mnt/Backup/Backup/  <<backup-opts()>> backup /home/yantar92; break;;
        [Nn]* ) break;;
        * ) echo "Please answer yes or no.";;
    esac
done
while true; do
    read -p "Check integrity of the local backup? " yn
    case $yn in
        [Yy]* ) restic -r /mnt/Backup/Backup/ check; break;;
        [Nn]* ) break;;
        * ) echo "Please answer yes or no.";;
    esac
done
# Sync with remote
while true; do
    read -p "Sync with remote backup? " yn
    case $yn in
        [Yy]* ) restic -r rclone:backup:Backup <<backup-opts()>> --password-command "gpg -q  --for-your-eyes-only -d /home/yantar92/.password-store/restic-remote.gpg" backup /home/yantar92/; break;;
        [Nn]* ) break;;
        * ) echo "Please answer yes or no.";;
    esac
done
while true; do
    read -p "Check integrity of the remote repository? " yn
    case $yn in
        [Yy]* ) restic -r rclone:backup:Backup --password-command "gpg -q  --for-your-eyes-only -d /home/yantar92/.password-store/restic-remote.gpg" check; break;;
        [Nn]* ) break;;
        * ) echo "Please answer yes or no.";;
    esac
done
# Upload local archive to remote
while true; do
    read -p "Backup archive to remote backup? " yn
    case $yn in
        [Yy]* ) restic -r rclone:backup:Archive --password-command "gpg -q  --for-your-eyes-only -d /home/yantar92/.password-store/restic-remote.gpg" backup /mnt/Backup/Archive/; break;;
        [Nn]* ) break;;
        ,,* ) echo "Please answer yes or no.";;
    esac
done
while true; do
    read -p "Check integrity of the remote repository? " yn
    case $yn in
        [Yy]* ) restic -r rclone:backup:Archive --password-command "gpg -q  --for-your-eyes-only -d /home/yantar92/.password-store/restic-remote.gpg" check; break;;
        [Nn]* ) break;;
        ,,* ) echo "Please answer yes or no.";;
    esac
done
# while true; do
#     read -p "Copy archive to remote backup? " yn
#     case $yn in
#         [Yy]* ) rclone copy --progress /mnt/Backup/Archive/ archive:; break;;
#         [Nn]* ) break;;
#         * ) echo "Please answer yes or no.";;
#     esac
# done
# Compress Org git repo (it tends to generate a huge extra space otherwise)
cd /home/yantar92/Org/; git maintenance run;
backup-mount.sh
/home/yantar92/.local/bin/backup-local.sh
/home/yantar92/.local/bin/backup-archive-to-remote.sh
backup-umount.sh
/home/yantar92/.local/bin/backup-remote.sh

[2020-12-22 Tue] trying encrypted backup using restic.

Mounting the backup drive

-----BEGIN PGP MESSAGE-----

hQEMA3SKBiQ2zhL6AQf/Zg/oCwC5vHX0eHMt0Z05J+zfNkTjU2bNRQy7GUsGQ0X1 qkGtuXW1YrFHMf7De6ttjghPpOQtsvpS2uIJVTBJi1GY8JhOc1Q3VMqfFMMoQvmT C8hVNOafV+nmwX88SiBVEHp9XD5R+z9tU8JyNPZkcEpR3OiezBuYqpjEtu4H+MSe Sq4jZX/RXDXLU8uNQ2XS9OstHzxMzDHE3OMMNUhbKv55Bl8YhqGQjwuYsSdORfy5 dBOlFKdKJZ3YjaM6K7vFhTA9f7imfxivPdN+xHu+HEgGYBgFmF8oEszoKPNC3u8K wwxmhtWpMveXcOyRs0ePeMcrBTa13pG/x4gU5PpAZNLArwGTn2oYdREG6irl36Lj SrnAULg7bkFTpdpdNIZVXtSBsDWyUTSnSvS6IOEGYq8S88B4oqj9kvcqHbwpC04v Dpk29hX83E7R6mbX813SYOczLGWAL6UnINLF6OrJBZ8HaAeAVxXNboDX6O5C/9RQ ss8y+b9rJxX5FpM5fOVjdr7SnFItkw1fOgWE6VRkCrD5HqyM5b3W8kVNRSCM29ub V7rrNTcXg0XCCWGB7qtM1Y2VNGM27jKiGFu8nclGHO3Xydo5nlPnGd0AZgCiMaoF UITWT+SffEXRUAz9VGMHLwuQxXI4woXoXPc85QHHVR0DF2ZLIFxuEw2S6mE3EJSE SL7r0a9vkFdu79qaneGzi9D6QGrsYspVB7NuZF9n9uof24LpHHT3Mf2ivL0eo0ze bgL17LjBsUi92nSuyI8GukbvDiHSl2B/aOaHWMIWLnjsFljnPq77DYQO0DywJKFW uQrUDzTdIDZ3g/A4Ge3FAd4= =Zsur -----END PGP MESSAGE-----

Mounting Nextcloud via davfs

Need net-fs/davfs2
net-fs/davfs2

Syncing backup with remote via net-misc/rclone

net-misc/rclone

Midnight commander

app-misc/mc

Mostly used to access ftp.

Should figure out how to used diredEND

Screencasting

Using (pip) screenkey + media-video/peek

media-video/peek
pip install screenkey
Note the screenkey repoEND
media-video/peek ~amd64
screenkey &
peek;
ps aux | grep screenkey | awk '{print $2}' | xargs kill

Coolreader: reading fb2 and other ebooks

app-text/crqt-ng

pdf manipulation: pdftk

app-text/pdftk

xyscan: analyse scientific plot images to extract data

sci-visualization/xyscan

pandoc: convert between text and document formats

app-text/pandoc

app-text/djvu for rare cases when I need to read of convert djvu files

app-text/djvu

www-client/surfraw for Emacs helm integration

www-client/surfraw

Remap control, shift, and caps (using x11-aps/xmodmap)

x11-apps/xmodmap

[[id:Emacswiki_emacs_swap_contr_alt_and_caps_lockfef][[Emacswiki] EmacsWiki: Swap Control Alt And Caps Lock]]:

  • Caps_Lock <-> Control
xmodmap -pke | grep "$key" | awk '{print $2}'
! First clear all modifiers, caps lock & control
clear lock
clear control

keycode  66 = Control_L

! We need to set keycodes first, as some programs (emacs!) read
! the key names and seem to ignore what's below.
add control = Control_R Control_L
xmodmap ~/.xmodmap

x11-apps/mesa-progs: testing 3d acceleration

x11-apps/mesa-progs

app-text/languagetool: Grammar checking

app-text/languagetool
app-text/languagetool ~amd64

app-text/vale: Text linter

app-text/vale::guru
app-text/vale
StylesPath = styles

MinAlertLevel = suggestion

Packages = proselint, write-good, alex, Readability, Joblint

[*]
BasedOnStyles = Vale, proselint, write-good, alex, Readability, Joblint

media-gfx/graphviz: LaTeX for diagrams

media-gfx/graphviz
media-gfx/graphviz svg doc

app-text/diffpdf

app-text/diffpdf

Compare pdfs visually and/or textually

redshift: colour adjustment according to time of day

-----BEGIN PGP MESSAGE-----

hQEMA3SKBiQ2zhL6AQf/V0icLktLNAF/Ql1alL2d7W0miRni716mlXIyfXLxq3dT +RNDJWU4ThvreV5Er06BskFX2fTJlKG88pChA/+ZXrBNu3g2v1qrK8Zr/QpVXorr fUZvDV4Ax0Uu910thS1Z32nQM6ja2gANheVmODj3zA7gHGiwjW008ybtpLB0G2Zk qWHdzjwaL4G8O0Ynv6vkApx0a8o9oouFu0c376K/zuOUt+a31/4Yh+FhWh4PWOV9 tmlIUABVFh2bs9UqxdFtNSgoLcmoQ/kcTQeBo59lKbTgG9MzzKGrtq/GDOCd1Eyl W4lFiLbMWMKwln66J5b2ZRDybZEzQd3IgTO4wBbcxdLAHwEuMcj+MSolk2/VheP5 3jiAzimPzjzNb0GfdroM7BBDUtVjO+/g1Mq+O6okybeyOuIdpoEtn66xxvLfVmfv AUj21/haVSaLlvdbe3iDuIvgB0PrBzOFcM52ErEDITtd2+6Ey13UoKl7vwR7iV7P 4YdiyZQkK0Sa55y6vjg7yTADAK5LgieTS4Rzm+hSdJY+A8BsnsB3UxIGoUFTV0T8 Roy3SFfwnOsPnoV0WH8E0olKEJFLIbUP4POl877gwrJY0erHC5NGn4rwaY8QSxC6 46CT6Hal43NbXVZ3NZdImBU= =3ugs -----END PGP MESSAGE-----

sys-fs/avfs: browse archives

Required for Emacs Browse archives with =dired-avfs=

sys-fs/avfs

Python libs

pandas

dev-python/pandas
dev-python/pandas doc

numpy

dev-python/numpy
dev-python/numpy doc

matplotlib

dev-python/matplotlib

Games

Need media-video/rtmpdump for Loop hero

media-video/rtmpdump

obs-studio - recording

media-video/obs-studio
media-video/obs-studio v4l
media-video/obs-studio ~amd64

app-admin/killproc: For usleep

app-admin/killproc

app-portage/gentoolkit for equary and revdep-rebuild commands

app-portage/gentoolkit

app-portage/eix quickly search portage packages

app-portage/eix

perf: Performance debugging

dev-util/perf

app-dicts/wordnet package

app-dicts/wordnet

Libreoffice

app-office/libreoffice-bin

Firefox (needed for jitsi that does not work well with Qutebrowser)

www-client/firefox-bin

System packages for testing optional modules of Org

Lilypond

media-sound/lilypond

R

dev-lang/R

[2024-06-05 Wed] Need to compile image support to produce :results graphics in R src blocks.

dev-lang/R png jpg

git-annex

dev-vcs/git-annex

Maxima

sci-mathematics/maxima

Octave

sci-mathematics/octave

Julia

dev-lang/julia-bin

ESS

app-emacs/ess

Cider (clojure)

app-emacs/cider

Leiningen (clojure)

dev-java/leiningen-bin

Geiser (scheme)

app-emacs/geiser
app-emacs/geiser-guile

Repair corrupted git repositories

dev-vcs/git-repair

GDB

dev-debug/gdb

Plantuml

media-gfx/plantuml

Clojure

dev-lang/clojure
dev-java/leiningen-bin

Python for research

Setup python to use local venv by default

. ~/.venv/bin/activate
python "$@"
. ~/.venv/bin/activate
pyright "$@"
. ~/.venv/bin/activate
pyright-langserver "$@"

Pyton linters

dev-python/yapf
dev-python/rope
dev-python/flake8
dev-python/pylint

Python LSP

Use pyright. There is no ebuild, so I have to use local pip install.

Vesta

sci-chemistry/vesta-bin::guru
sci-chemistry/vesta-bin VESTA
sci-chemistry/vesta-bin

Tkinter GUI

dev-lang/python tk

Guix setup

Recoll - file system indexer

Searching across own notes is excellent, but it is sometimes also useful to index all the data, including pdfs and local web page snapshots when looking to some rarely accessed data.

app-misc/recoll

Zotero - collaboration

app-text/zotero-bin

Archived