UP | HOME

Holymacs ~ Jarvis module

Table of Contents

Helpers

(defun holymacs-random (items)
  "Pick a random element from a list"
  (nth (random (length items)) items))

(defun eval-replace-last-sexp ()
  (interactive)
  (let ((value (eval (preceding-sexp))))
    (kill-sexp -1)
    (insert (format "%S" value))))

Youtube

(defcustom jarvis/youtube-sources '()
  "List of some Youtube videos; `jarvis/play-youtube` will pick one of them, randomly, and open it"
  :type '(repeat (list :tag "sources" (string :tag "title") (string :tag "url"))))

(defun jarvis/play-youtube (&optional url)
  "Randomly pick one of the videos defined in `jarvis/youtube-sources` and then play it."
  (interactive (list (completing-read "URL: " jarvis/youtube-sources)))
  (let ((command (format "mpv \"%s\"" url))
        (buffer-name (format "mpv: %s" url)))
    (async-shell-command command buffer-name)))

(defun jarvis/open-random-youtube ()
  "Open in mpv one of the sources in `jarvis/youtube-sources` and then open it."
  (interactive)
  (let ((url (holymacs-random jarvis/youtube-sources)))
    (jarvis/open-youtube url)))

(defun jarvis/open-youtube (&optional url)
  "Open in browser one of the sources in `jarvis/youtube-sources`."
  (interactive (list (completing-read "URL: " jarvis/youtube-sources)))
  (browse-url-default-browser url))

Check documentation about Google APIs.

(use-package password-store) ;;  (password-store-get "youtube-api")
(use-package request
  :after password-store
  :config
  (defcustom jarvis/youtube-api-key "" "Youtube API key")
  (defvar jarvis/youtube-api-key "" "Youtube API key")
  (defun jarvis/channel-vids (url)
    "Get a list of videos given a Youtube channel (url)"
    (request "https://www.googleapis.com/youtube/v3/search"
      :params `(("key" . ,jarvis/youtube-api-key) ("maxResults" . 2))
      :parser 'json-read
      :success (cl-function
                (lambda (&key data &allow-other-keys)
                  (when data
                    (with-current-buffer (get-buffer-create "*youtube-vids*")
                      (erase-buffer)
                      (insert (prin1-to-string data))
                      (pop-to-buffer (current-buffer)))))))))

Alarm!

(use-package alarm-clock
  :config
  (defun holymacs/jarvis/set-alarm-10min-fire! ()
    (interactive)
    (alarm-clock-set "10min" "FIRE!"))
  :bind
  ("M-o ! 1" . #'holymacs/jarvis/set-alarm-10min-fire!))

News and Mail

RSS Reader (elfeed)

(use-package elfeed
  :after 'god-mode
  :bind
  (:map elfeed-search-mode-map ("U" . (lambda () (interactive) (run-at-time "10min" t #'elfeed-update))))
  :config
  (add-to-list 'god-exempt-major-modes 'elfeed-search-mode-map))

Enanche your elfeed experience with elfeed-tube! ❤️ Thanks a lot to karthink 🙏

(use-package elfeed-tube
  :after elfeed
  :demand t
  :config
  ;; (setq elfeed-tube-auto-save-p nil) ; default value
  ;; (setq elfeed-tube-auto-fetch-p t)  ; default value
  (elfeed-tube-setup)

  :bind (:map elfeed-show-mode-map
                ("F" . elfeed-tube-fetch)
                ([remap save-buffer] . elfeed-tube-save)
                :map elfeed-search-mode-map
                ("F" . elfeed-tube-fetch)
                ([remap save-buffer] . elfeed-tube-save)))

(use-package elfeed-tube-mpv
  :after elfeed-tube
  :bind (:map elfeed-show-mode-map
                ("C-c C-f" . elfeed-tube-mpv-follow-mode)
                ("C-c C-w" . elfeed-tube-mpv-where)))

Mu4e

Dependencies

Firstly, install mu and a MailBox <-> Imaps syncronizer like mbsync. You should also rely on pass, the the standard unix password manager, in order to store and fetch your passwords in a secure way.

Useful links:

Configuration

IMAPAccount gmail-imaps
Host imap.gmail.com
Port 993
#UseIMAPS yes
#RequireSSl yes
SSLType IMAPS
SSLVersions TLSv1.2
AuthMechs PLAIN
User your-email@gmail.com
#Pass supercomplicatedpassword
PassCmd "pass gmail/imaps"

IMAPStore gmail-remote
Account gmail-imaps

MaildirStore gmail-local
Path ~/MailDir/gmail/
Inbox ~/MailDir/gmail/INBOX
Subfolders Verbatim

Channel gmail
#Master :gmail-remote:
#Slave :gmail-local:
Far :gmail-remote:
Near :gmail-local:
Create Both
Expunge Both
Patterns *
SyncState *
(require 'mu4e)
(setq send-mail-function 'smtpmail-send-it
	message-send-mail-function 'smtpmail-send-it
	mu4e-get-mail-command "mbsync -a"
	;;mu4e-maildir "~/Mail"
	mu4e-update-interval (* 60 10)
	smtpmail-default-smtp-server "smtp.gmail.com"
	smtpmail-smtp-server "smtp.gmail.com"
	;;smtpmail-stream-type 'ssl
	;;smtpmail-smtp-service 465
	smtpmail-stream-type 'starttls
	smtpmail-smtp-service 587
	;;smtpmail-use-gnutls t
	mu4e-drafts-folder "/[Gmail].Drafts"
	mu4e-sent-folder   "/[Gmail].Sent Mail"
	mu4e-trash-folder  "/[Gmail].Trash"
	mm-discouraged-alternatives '("text/html")
	mml-secure-openpgp-signers '("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"))
(add-hook 'message-send-hook 'mml-secure-message-sign-pgpmime)
(auth-source-pass-enable)
(setq smtpmail-debug-info t)

To set multiple accounts within mu4e you have to set the mu4e-contexts variable. An interesting screencast/article from System Crafters about that can be found here.

(setq mu4e-contexts
	(list
	 ;; Work account
	 (make-mu4e-context
	  :name "GMail"
	  :match-func
	  (lambda (msg)
	    (when msg
	      (string-prefix-p "/Gmail" (mu4e-message-field msg :maildir))))
	  :vars '((user-mail-address . "my-email@gmail.com")
		  (user-full-name    . "My-Email (Gmail)")
		  (mu4e-drafts-folder  . "/Gmail/[Gmail]/Drafts")
		  (mu4e-sent-folder  . "/Gmail/[Gmail]/Sent Mail")
		  (mu4e-refile-folder  . "/Gmail/[Gmail]/All Mail")
		  (mu4e-trash-folder  . "/Gmail/[Gmail]/Trash")))

	 ;; Personal account
	 (make-mu4e-context
	  :name "Yahoo"
	  :match-func
	  (lambda (msg)
	    (when msg
	      (string-prefix-p "/Yahoo" (mu4e-message-field msg :maildir))))
	  :vars '((user-mail-address . "my-email@yahoo.com")
		  (user-full-name    . "My-Email (Yahoo)")
		  (mu4e-drafts-folder  . "/Yahoo/Drafts")
		  (mu4e-sent-folder  . "/Yahoo/Sent")
		  (mu4e-refile-folder  . "/Yahoo/Archive")
		  (mu4e-trash-folder  . "/Yahoo/Trash")))))

Tweaking

Mu4e does not allow to save all attachments in one shot (some workarounds to achieve that involve completion frameworks like Ivy or use Embark to select all attachments from the minibuffer after invoking mu4e-view-save-attachment). For more details see this issue on Github.

Media

(use-package bongo)

cht.sh

curl https://cht.sh 
(defun cht.sh-help (helpable)
  (interactive "sHelp: ")
  (cht.sh-query "/:help"))

(defun cht.sh-list ()
  (interactive)
  (cht.sh-query "/:list"))

(defun cht.sh-query (query)
  (interactive "sQuery: ")
  (let ((cmd (format "curl https://cht.sh/%s" query))
        (buffer-name (format "*cht-sh* <%s>" query)))
    (async-shell-command cmd buffer-name)))

Search on the web: engine-mode

(use-package engine-mode
  :init
  (setq engine/browser-function #'browse-url-firefox)
  :config
  (defun engine/execute-search (search-engine-url browser-function search-term)
    "Display the results of the query."
    (interactive)
    (let ((browse-url-handlers nil)
            (browse-url-browser-function browser-function))
        (browse-url
         (format-spec search-engine-url
                      (format-spec-make ?s (url-hexify-string search-term))))))

  (defengine dropbox
    "https://www.dropbox.com/search/personal?query=%s"
    :browser 'browse-url-firefox
    :keybinding "D")

  (defengine duckduckgo
    "https://duckduckgo.com/?q=%s"
    :browser 'eww-browse-url
    :keybinding "d")

  (defengine github
    "https://github.com/search?ref=simplesearch&q=%s"
    :keybinding "h")

  (defengine google
    "https://www.google.com/search?ie=utf-8&oe=utf-8&q=%s"
    :browser 'browse-url-firefox
    :keybinding "g")

  (defengine google-images
    "https://www.google.com/images?hl=en&source=hp&biw=1440&bih=795&gbv=2&aq=f&aqi=&aql=&oq=&q=%s"
    :keybinding "i")

  (defengine google-maps
    "https://maps.google.com/maps?q=%s"
    :docstring "Google Maps"
    :keybinding "m")

  (defengine google-maps
    "https://www.openstreetmap.org/search?query=%s"
    :docstring "OpenStreetMap"
    :keybinding "M")

  (defengine google-translate-to-it
    "https://translate.google.it/?sl=auto&tl=it&text=%s&op=translate"
    :docstring "Translate to IT"
    :keybinding "t")

  (defengine google-translate-to-en
    "https://translate.google.it/?sl=it&tl=en&text=%s&op=translate"
    :docstring "Translate from IT to English"
    :browser 'browse-url-firefox
    :keybinding "T")

  (defengine stack-overflow
    "https://stackoverflow.com/search?q=%s"
    :keybinding "s")

  (defengine wikipedia
    "https://www.wikipedia.org/search-redirect.php?language=it&go=Go&search=%s"
    :keybinding "w"
    :docstring "Searchin' the wikis."
    :browser 'eww-browse-url)

  (defengine youtube
    "https://www.youtube.com/results?aq=f&oq=&search_query=%s"
    :keybinding "y")

  (defengine amazon
    "https://www.amazon.it/s/ref=nb_sb_noss?&field-keywords=%s"
    :browser 'browse-url-firefox
    :keybinding "a")

  (defengine cap
    "https://www.nonsolocap.it/cap?k=%s&c=pescara"
    :browser 'browse-url-firefox
    :keybinding "C")

  (engine-mode t))

Date: 2024-04-20 Sat 00:00

Emacs 30.1 (Org mode 9.7.11)

Validate