2019-10-02 16:31:34 +02:00
|
|
|
(defun eval-in-emacs (&rest s-exps)
|
|
|
|
"Evaluate S-exps with `emacsclient'."
|
2019-12-09 15:35:57 +01:00
|
|
|
(let ((s-exps-string (cl-ppcre:regex-replace-all
|
2019-10-02 16:31:34 +02:00
|
|
|
;; Discard the package prefix.
|
2019-12-09 15:35:57 +01:00
|
|
|
"next-user::?"
|
2019-10-02 16:31:34 +02:00
|
|
|
(write-to-string
|
2019-12-09 15:35:57 +01:00
|
|
|
`(progn ,@s-exps) :case :downcase)
|
|
|
|
"")))
|
|
|
|
(log:debug "Sending to Emacs: ~s" s-exps-string)
|
2019-10-02 16:31:34 +02:00
|
|
|
(ignore-errors (uiop:run-program
|
|
|
|
(list "emacsclient" "--eval" s-exps-string)))))
|
|
|
|
|
2020-03-31 10:13:04 +02:00
|
|
|
(defvar *my-keymap* (keymap:make-keymap "my-map")
|
2019-10-02 16:31:34 +02:00
|
|
|
"Keymap for `my-mode'.")
|
|
|
|
|
|
|
|
(define-command org-capture (&optional (buffer (current-buffer)))
|
|
|
|
"Org-capture current page."
|
|
|
|
(eval-in-emacs
|
|
|
|
`(org-link-set-parameters
|
|
|
|
"next"
|
|
|
|
:store (lambda ()
|
|
|
|
(org-store-link-props
|
|
|
|
:type "next"
|
|
|
|
:link ,(url buffer)
|
|
|
|
:description ,(title buffer))))
|
|
|
|
`(org-capture)))
|
2020-03-31 10:13:04 +02:00
|
|
|
(define-key *my-keymap* "C-M-o" #'org-capture)
|
2019-10-02 16:31:34 +02:00
|
|
|
|
|
|
|
(define-command youtube-dl-current-page (&optional (buffer (current-buffer)))
|
|
|
|
"Download a video in the currently open buffer."
|
|
|
|
(eval-in-emacs
|
|
|
|
(if (search "youtu" (url buffer))
|
|
|
|
`(progn (youtube-dl ,(url buffer)) (youtube-dl-list))
|
|
|
|
`(ambrevar/youtube-dl-url ,(url buffer)))))
|
2020-03-31 10:13:04 +02:00
|
|
|
(define-key *my-keymap* "C-M-c d" #'youtube-dl-current-page)
|
2019-10-02 16:31:34 +02:00
|
|
|
|
|
|
|
(define-command play-video-in-current-page (&optional (buffer (current-buffer)))
|
|
|
|
"Play video in the currently open buffer."
|
|
|
|
(uiop:run-program (list "mpv" (url buffer))))
|
2020-03-31 10:13:04 +02:00
|
|
|
(define-key *my-keymap* "C-M-c v" #'play-video-in-current-page)
|
2019-10-02 16:31:34 +02:00
|
|
|
|
|
|
|
(define-mode my-mode ()
|
|
|
|
"Dummy mode for the custom key bindings in `*my-keymap*'."
|
2020-04-26 22:12:33 +02:00
|
|
|
((keymap-scheme :initform (keymap:make-scheme
|
|
|
|
scheme:emacs *my-keymap*
|
|
|
|
scheme:vi-normal *my-keymap*))))
|
2019-10-02 16:31:34 +02:00
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
2019-09-24 17:28:09 +02:00
|
|
|
(defvar +youtube-dl-command+ "youtube-dl"
|
|
|
|
"Path to the 'youtube-dl' program.")
|
|
|
|
|
|
|
|
(defun auto-yt-dl-handler (url)
|
|
|
|
"Download a Youtube URL asynchronously to /tmp/videos/.
|
|
|
|
Videos are downloaded with `+youtube-dl-command+'."
|
|
|
|
(let ((uri (quri:uri url)))
|
|
|
|
(when (and uri
|
|
|
|
(member-string (quri:uri-domain uri) '("youtube.com" "youtu.be"))
|
|
|
|
(string= (quri:uri-path uri) "/watch"))
|
|
|
|
(log:info "Youtube: downloading ~a" url)
|
|
|
|
(uiop:launch-program (list +youtube-dl-command+ url "-o" "/tmp/videos/%(title)s.%(ext)s"))))
|
|
|
|
url)
|
|
|
|
|
|
|
|
(defun old-reddit-handler (url)
|
2019-09-10 17:15:03 +02:00
|
|
|
(let* ((uri (quri:uri url)))
|
2020-03-14 16:35:43 +01:00
|
|
|
(if (search "reddit.com" (quri:uri-host uri))
|
2019-09-10 17:15:03 +02:00
|
|
|
(progn
|
|
|
|
(setf (quri:uri-host uri) "old.reddit.com")
|
|
|
|
(let ((new-url (quri:render-uri uri)))
|
|
|
|
(log:info "Switching to old Reddit: ~a" new-url)
|
|
|
|
new-url))
|
|
|
|
url)))
|
2019-09-24 17:28:09 +02:00
|
|
|
|
|
|
|
(defvar *my-unproxied-domains*
|
|
|
|
'("jit.si"
|
2019-10-03 15:55:42 +02:00
|
|
|
"steampowered.com"))
|
2019-09-24 17:28:09 +02:00
|
|
|
|
|
|
|
(defun auto-proxy-handler (url)
|
|
|
|
(let* ((uri (quri:uri url))
|
|
|
|
(domain (and uri (quri:uri-domain uri))))
|
2019-11-17 14:01:06 +01:00
|
|
|
;; TODO: Turn on/off proxy, not mode.
|
2019-09-24 17:28:09 +02:00
|
|
|
(when domain
|
|
|
|
(next/proxy-mode:proxy-mode
|
|
|
|
:activate
|
|
|
|
(not (member-string domain *my-unproxied-domains*)))))
|
|
|
|
url)
|
|
|
|
|
2019-10-02 16:31:34 +02:00
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
(defvar *my-blocked-hosts*
|
|
|
|
(next/blocker-mode:make-hostlist
|
|
|
|
:hosts '("platform.twitter.com"
|
|
|
|
"syndication.twitter.com"
|
|
|
|
"m.media-amazon.com")))
|
|
|
|
|
2020-04-21 19:54:48 +02:00
|
|
|
(define-configuration next/blocker-mode:blocker-mode
|
|
|
|
((next/blocker-mode:hostlists (append (list *my-blocked-hosts*) %slot-default))))
|
|
|
|
|
|
|
|
;; (define-mode my-blocker-mode (next/blocker-mode:blocker-mode)
|
|
|
|
;; "Blocker mode with custom hosts from `*my-blocked-hosts*'."
|
|
|
|
;; ((next/blocker-mode:hostlists :initform (list *my-blocked-hosts* next/blocker-mode:*default-hostlist*))))
|
2019-10-02 16:31:34 +02:00
|
|
|
|
2020-04-15 11:51:50 +02:00
|
|
|
(define-configuration buffer
|
|
|
|
((default-modes (append
|
|
|
|
'(my-mode vi-normal-mode
|
2020-04-21 19:54:48 +02:00
|
|
|
;; my-blocker-mode
|
|
|
|
blocker-mode
|
2020-04-15 11:51:50 +02:00
|
|
|
proxy-mode)
|
2020-04-21 19:54:48 +02:00
|
|
|
%slot-default))
|
2020-04-15 11:51:50 +02:00
|
|
|
(load-hook (hooks:make-hook-string->string
|
|
|
|
:handlers (list #'old-reddit-handler
|
|
|
|
#'auto-proxy-handler)
|
|
|
|
:combination #'hooks:combine-composed-hook))))
|
2019-10-02 16:31:34 +02:00
|
|
|
|
2020-04-08 15:41:14 +02:00
|
|
|
(defmacro with-maybe-gpg-file ((stream filespec) &body body)
|
|
|
|
"Evaluate BODY with STREAM bound to FILESPEC.
|
|
|
|
FILESPEC may be a GPG-encrypted file."
|
|
|
|
`(let ((clear-data (handler-case
|
|
|
|
(with-output-to-string (out)
|
|
|
|
(uiop:run-program (list "gpg" "-d" ,filespec) :output out))
|
|
|
|
(error ()
|
|
|
|
(alexandria:read-file-into-string ,filespec)))))
|
|
|
|
(with-input-from-string (,stream clear-data)
|
|
|
|
,@body)))
|
|
|
|
|
|
|
|
(defun format-c->lisp (s)
|
|
|
|
"Incomplete substitution of C format string to Lisp format string.
|
|
|
|
Recognized formats:
|
|
|
|
- %%
|
|
|
|
- %s"
|
|
|
|
(str:join "%" (mapcar (lambda (s) (str:replace-all "%s" "~a" s))
|
|
|
|
(str:split "%%" s))))
|
|
|
|
|
|
|
|
(defun read-emacs-engines (stream)
|
|
|
|
"Return a list of (NAME URL SHORTCUT)."
|
|
|
|
(loop for object = (read stream nil :eof)
|
|
|
|
until (eq object :eof)
|
|
|
|
when (eq (car object) 'defengine)
|
|
|
|
collect (list ;; (nth 1 object) ; No need for name?
|
|
|
|
(getf (nthcdr 3 object) :keybinding)
|
|
|
|
(format-c->lisp (nth 2 object)))))
|
|
|
|
|
2020-02-22 15:39:48 +01:00
|
|
|
(defvar my-search-engines
|
2020-04-08 15:41:14 +02:00
|
|
|
(loop for file in '("~/.emacs.d/lisp/init-engine.el"
|
|
|
|
"~/personal/bookmarks/engines.el")
|
|
|
|
append (with-maybe-gpg-file (s file)
|
|
|
|
(read-emacs-engines s))))
|
2020-02-22 15:39:48 +01:00
|
|
|
|
2020-04-15 11:51:50 +02:00
|
|
|
(define-configuration browser
|
2020-04-26 22:12:55 +02:00
|
|
|
((search-engines (append my-search-engines %slot-default))
|
|
|
|
(session-restore-prompt :always-restore)))
|
2019-10-02 16:31:34 +02:00
|
|
|
|
|
|
|
(defmethod deserialize-bookmarks (stream)
|
|
|
|
"This version of deserialize-bookmarks is compatibly with Ambrevar's EWW
|
|
|
|
format."
|
|
|
|
(handler-case
|
|
|
|
(let ((*standard-input* stream))
|
|
|
|
(let ((entries (read stream)))
|
|
|
|
(mapcar (lambda (entry)
|
|
|
|
(when (getf entry :date)
|
|
|
|
(setf (getf entry :date)
|
|
|
|
(local-time:parse-timestring (getf entry :date))))
|
|
|
|
(when (getf entry :time)
|
|
|
|
(let ((timestamp (asctime->timestamp (getf entry :time))))
|
|
|
|
(when timestamp
|
|
|
|
(setf (getf entry :date) timestamp)))
|
|
|
|
(remf entry :time))
|
|
|
|
(when (getf entry :search)
|
|
|
|
(setf (getf entry :search-url) (getf entry :search))
|
|
|
|
(remf entry :search))
|
|
|
|
(when (getf entry :mark)
|
|
|
|
(setf (getf entry :shortcut) (getf entry :mark))
|
|
|
|
(remf entry :mark))
|
|
|
|
(apply #'make-instance 'bookmark-entry
|
|
|
|
entry))
|
|
|
|
entries)))
|
|
|
|
(error (c)
|
|
|
|
(log:error "Error during bookmark deserialization: ~a" c)
|
|
|
|
nil)))
|
2020-02-22 15:39:48 +01:00
|
|
|
|
|
|
|
(setf next/vcs:*vcs-projects-roots* '("~/projects"
|
|
|
|
"~/common-lisp"
|
|
|
|
"~/.local/share/emacs/site-lisp"))
|
2020-04-21 19:54:48 +02:00
|
|
|
|
|
|
|
(defun my-format-status (window)
|
|
|
|
(declare (ignore window))
|
|
|
|
(let* ((buffer (current-buffer))
|
2020-04-22 21:17:11 +02:00
|
|
|
(buffer-count (1+ (or (position buffer
|
|
|
|
(sort (alexandria:hash-table-values (buffers *browser*))
|
|
|
|
#'<
|
|
|
|
:key #'id))
|
|
|
|
0))))
|
2020-04-26 22:13:17 +02:00
|
|
|
(str:concat
|
|
|
|
(markup:markup
|
|
|
|
(:b (format nil "[~{~a~^ ~}]"
|
|
|
|
(mapcar (lambda (m) (str:replace-all "-mode" ""
|
|
|
|
(str:downcase
|
|
|
|
(class-name (class-of m)))))
|
|
|
|
(modes buffer)))))
|
|
|
|
(format nil " (~a/~a) ~a — ~a"
|
|
|
|
buffer-count
|
|
|
|
(hash-table-count (buffers *browser*))
|
|
|
|
(url buffer)
|
|
|
|
(title buffer)))))
|
2020-04-21 19:54:48 +02:00
|
|
|
|
|
|
|
(define-configuration window
|
|
|
|
((status-formatter #'my-format-status)))
|
2020-04-26 22:13:17 +02:00
|
|
|
|