Skip to content
Find file
Fetching contributors…
Cannot retrieve contributors at this time
402 lines (346 sloc) 14.4 KB
;; eslide
; (setq eslide-edit-mode-map (make-sparse-keymap))
;(set-fontset-font "fontset-startup" '(#x00ff . #xffff) "Sazanami Mincho" nil 'prepend)
(defgroup eslide nil "Eslide")
(defvar eslide-control-map
(let ((m (make-sparse-keymap)))
(loop for (key . command) in
`(("n" . eslide-next)
("SPC" . eslide-next)
("<right>" . eslide-next)
("p" . eslide-prev)
("DEL" . eslide-prev)
("<left>" . eslide-prev)
("<prior>" . ,(lambda () (interactive) (scroll-down 1)))
("<next>" . ,(lambda () (interactive) (scroll-up 1)))
("+" . eslide-text-scale-increase)
("=" . eslide-text-scale-increase)
("<XF86AudioRaiseVolume>" . eslide-text-scale-increase)
("-" . eslide-text-scale-decrease)
("_" . eslide-text-scale-decrease)
("<XF86AudioLowerVolume>" . eslide-text-scale-decrease)
do (define-key m (kbd key) command))
(defconst +eslide-separator+ "^----\n")
(defconst +code-start+ " ")
(defvar eslide-slide-source nil)
(defvar eslide-current-slide-overlay nil)
(defvar eslide-start-time nil)
(defvar eslide-fixed-overlays nil)
(defvar eslide-current-subtitle "")
(defvar eslide-buffers nil)
(defface eslide-code '((t
(:underline "green" :background "grey20"))) "indented code face")
(defface eslide-subtitle '((t (:foreground "yellow" :height 200))) "subtitle face")
(defun eslide-post-command-hook ()
(if (equal (buffer-name (current-buffer)) "*ESlide Show*")
(internal-show-cursor nil nil)
(internal-show-cursor nil t)))
(add-hook 'post-command-hook #'eslide-post-command-hook)
(defun eslide-ensure-buffers (&optional rebuild)
"Setup the variable `eslide-buffers'.
Optional argument REBUILD controls rebuilding buffers even when they already exist."
(when (not (reduce (lambda (&optional a b) (and (boundp a) (boundp b) a b))
(loop for buf in eslide-buffers collect (buffer-live-p buf))))
(setf eslide-buffers nil))
(when (or (not eslide-buffers) rebuild)
(setf eslide-buffers
(loop for i in (list (get-buffer-create "*ESlide Show*")
(get-buffer-create "*ESlide Notes*"))
do (with-current-buffer i
(use-local-map eslide-control-map)
(setq buffer-read-only t))
collect i))
(with-current-buffer (car eslide-buffers)
(variable-pitch-mode 1))))
(defun eslide-buffers (&optional which) ;; the which is a lie.
"Return all eslide buffers.
Optional argument WHICH controls which buffers to return and in what order; currently \"show\" and \"notes\" are known."
(defun eslide-notes ()
(cadr eslide-buffers))
(defun eslide-show ()
(car eslide-buffers))
(defmacro build-with-<foo>-macro (macro-name buffer-getter-form)
`(defmacro ,macro-name (&body body)
`(with-current-buffer ,,buffer-getter-form ,@body)))
(defun eslide-start nil
"Start the slideshow using the current buffer as the slide source."
(setq eslide-slide-source (current-buffer))
(goto-char (point-min))
(when eslide-current-slide-overlay
(delete-overlay eslide-current-slide-overlay))
(setq eslide-current-slide-overlay
(make-overlay (point-min) (point-min) eslide-slide-source))
(overlay-put eslide-current-slide-overlay 'face '(:background "#040"))
(setq eslide-start-time (cadr (current-time)))
(defvar eslide-keywords `(,+eslide-separator+))
(define-derived-mode eslide-edit-mode fundamental-mode "ESlide[Edit]"
"Edit eslides"
(setq font-lock-defaults '(eslide-keywords nil ))
(define-key eslide-edit-mode-map (kbd "C-c C-c") #'eslide-start)
(define-key eslide-edit-mode-map (kbd "C-c C-n") #'eslide-next)
(define-key eslide-edit-mode-map (kbd "C-c C-p") #'eslide-prev))
(add-to-list 'auto-mode-alist (cons ".esl$" 'eslide-edit-mode))
(defun eslide-get-slide-near (pos)
(goto-char pos)
(let ((start (save-excursion
(or (and (re-search-backward +eslide-separator+ nil t) (match-end 0))
(end (save-excursion
(or (and (re-search-forward +eslide-separator+ nil t) (match-beginning 0))
(cons start end))))
(defun eslide-update-current-slide (&optional pos)
(with-current-buffer eslide-slide-source
(goto-char (or pos (overlay-start eslide-current-slide-overlay)))
(skip-chars-forward "-")
(destructuring-bind (start . end)
(eslide-get-slide-near (point))
(move-overlay eslide-current-slide-overlay start end)))))
(defun eslide-get-current-slide-text nil
(with-current-buffer eslide-slide-source
(overlay-start eslide-current-slide-overlay)
(overlay-end eslide-current-slide-overlay))))
(defun eslide--append-to-buffer (buffer format &rest args)
"Add message to the end of BUFFER"
(with-current-buffer buffer
(goto-char (point-max))
(let ((inhibit-read-only t))
(insert (apply #'format (cons (concat format "\n") args))))
(goto-char (point-max)))))
(defun eslide-show-note (format &rest args)
(destructuring-bind (show notes) (eslide-buffers '(show notes))
(apply #'eslide--append-to-buffer (list* notes format args))
(with-current-buffer notes
(loop for win in (get-buffer-window-list notes nil t)
do (set-window-point win (point-max)))
(goto-char (point-max))
(recenter -1))))
(defun eslide-show-slide (text)
(with-current-buffer (eslide-show)
(let ((inhibit-read-only t))
(setq cursor-type nil)
(setq cursor-in-non-selected-windows nil)
(delete-region (point-min) (point-max))
(insert text)
(text-scale-increase 0)
(text-scale-increase (eslide-scale-for-slide text))
(when (not (eslide-no-lines-wrap-p (selected-window)))
(text-scale-decrease 1))
(goto-char (point-min)))))
(defmacro with-string-buffer (string &rest forms)
(insert ,string)
(unwind-protect (progn ,@forms)
(set-buffer-modified-p nil))
(buffer-substring (point-min) (point-max))))
(defun eslide-format-note (note face)
(propertize note 'face (list :inherit face))
(goto-char (point-min))
(insert "\n")
(while (< (point) (point-max))
(insert " ")
(defun eslide-fixed-overlay (start end)
"Add an overlay to make the font fixed width.
Argument START sdfsdf.
Argument END sdfasdf."
(let ((o (make-overlay start end (eslide-show))))
(push o eslide-fixed-overlays)
(overlay-put o 'face '(:family "DejaVu Sans Mono"))))
(defun fontify-code (code)
"Fontify CODE block and return result."
(with-string-buffer code
(setq buffer-file-name "/tmp/")
(setq buffer-file-name nil)))
(defun eslide-code-line-p ()
"Determine if the current line is a code line."
(looking-at +code-start+)))
(defun eslide-subtitle-line-p ()
"Determine if the current line is a subtitle line."
(looking-at "s:")))
(defun eslide-format-one-chunk ()
"Starting with the current line, format in place."
(cond ((eslide-code-line-p)
(let* ((start (line-beginning-position))
(end (or (apply #'max (loop collect (line-end-position)
while (and (eslide-code-line-p) (= 0 (forward-line)))))
(code (buffer-substring start end))
(fontified-code (fontify-code code)))
(goto-char start)
(delete-region start end)
(narrow-to-region start (point))
(goto-char (point-min))
(insert fontified-code)
(goto-char (point-min))
(while (re-search-forward (concat "^" +code-start+) nil t)
(replace-match " "))
(eslide-fixed-overlay (point-min) (point-max))
(goto-char (point-max)))))
(setf eslide-current-subtitle
(concat eslide-current-subtitle " "
(+ (line-beginning-position) 2) ;; the 3 is "s:"
(delete-region (line-beginning-position) (line-end-position))
(when (looking-at "\n") (delete-char 1)))
(t nil)))
(defun eslide-format-slide ()
"Format the slide."
(goto-char (point-min))
(loop do (eslide-format-one-chunk) while (= 0 (forward-line)))
(when (> (length eslide-current-subtitle) 1)
(insert (propertize eslide-current-subtitle 'face '(eslide-subtitle)))
(defun eslide-cleanup-from-last-slide ()
"Cleanup globals."
;; kill overlays
(loop for o in (pop eslide-fixed-overlays) do (delete-overlay o))
(setf eslide-fixed-overlays nil)
;; kill subtitles
(setf eslide-current-subtitle ""))
(defun eslide-move (direction)
"Move next slide in positive or negative DIRECTION."
(with-current-buffer eslide-slide-source
;; find slide
(cond ((< direction 0)
(goto-char (overlay-start eslide-current-slide-overlay))
(ignore-errors (skip-chars-backward "-\n")))
((> direction 0)
(goto-char (overlay-end eslide-current-slide-overlay))
(ignore-errors (skip-chars-forward "-\n"))))
(eslide-update-current-slide (point)))
(with-current-buffer (eslide-notes)
(let ((inhibit-read-only t))
(delete-region (point-min) (point-max))))
;; show time in notes window
(eslide-show-note "time: %s"
(format "%d" (/ (- (cadr (current-time)) eslide-start-time) 60))
;; show current/next slide
(let ((text (eslide-get-current-slide-text)))
(eslide-show-note "current: %s"
(eslide-format-note text
(eslide-show-slide text))
(let ((next
(destructuring-bind (start . end)
(with-current-buffer eslide-slide-source
(goto-char (overlay-end eslide-current-slide-overlay))
(skip-chars-forward "-\n ")
(buffer-substring-no-properties start end))))
(eslide-show-note "next: %s"
(eslide-format-note next
(defun eslide-next nil
(eslide-move 1))
(defun eslide-prev nil
(eslide-move -1))
(defun eslide-text-scale-increase nil
(with-current-buffer (eslide-show)
(text-scale-increase 1)))
(defun eslide-text-scale-decrease nil
(with-current-buffer (eslide-show)
(text-scale-decrease 1)))
(defun eslide-minimize (list function)
"Return the element of LIST that FUNCTION makes smallest."
(caar (sort (mapcar (lambda (x) (cons x (funcall function x))) list)
(lambda (a b) (< (cdr a) (cdr b))))))
(defun eslide-narrowest-window (buffer)
(eslide-minimize (get-buffer-window-list buffer nil t)
(defun eslide-shortest-window (buffer)
(eslide-minimize (get-buffer-window-list buffer nil t)
(defun eslide-text-scale (size)
(text-scale-increase 0)
(text-scale-increase size))
(defmacro* with-eslide-temp-show-buffer (&body forms)
"Execute FORMS in a temp buffer (with special eslide settings in effect)."
(declare (indent 0))
`(with-selected-window (eslide-narrowest-window (eslide-show))
(variable-pitch-mode 1)
(defun eslide-buffer-fits-on-one-screen-p (window)
"Return T if current buffer when displayed on WINDOW will not need to scroll."
(set-window-buffer window (current-buffer))
(ignore-errors (while t (scroll-down)))
(goto-char (point-min))
(block nil
(condition-case e (scroll-up)
(error (return t)))
(return nil))))
(defun eslide-no-lines-wrap-p (window)
"Return T if no lines in current buffer wrap when displayed in WINDOW."
(with-selected-window window
;; (message "scale %d lines %d screen %d"
;; text-scale-mode-amount
;; (count-lines (point-min) (point-max))
;; (count-screen-lines (point-min) (point-max) nil window))
(= (count-lines (point-min) (point-max))
(count-screen-lines (point-min) (point-max) nil window))))
(defun* eslide-maximizing-text-scale (predicate window &optional (max 20))
"Increase text-scale until PREDICATE returns NIL, return max scale where PREDICATE returned nil. Operate on text of BUFFER as if it were displayed in WINDOW."
(cond ((not (window-live-p window)) 0)
(t (or (loop for scale from 0 to max
do (eslide-text-scale scale)
if (not (funcall predicate window))
return (1- scale)) max))))
(defun eslide-scale-for-slide (slide)
"Return the maximum text-scale value for SLIDE where no lines wrap and where every line is visible in every ESlide Show window."
(insert slide)
(goto-char (point-min))
(loop do (progn (goto-char (line-beginning-position))
(when (looking-at "s:")
(kill-line 2)))
while (zerop (forward-line)))
(let* ((show (eslide-show))
(narrowest (eslide-narrowest-window show))
(shortest (eslide-shortest-window show))
(max-size-width (eslide-maximizing-text-scale #'eslide-no-lines-wrap-p narrowest))
(max-size-height (eslide-maximizing-text-scale #'eslide-buffer-fits-on-one-screen-p shortest max-size-width)))
;; (message "size %d %d" max-size-width max-size-height)
(provide 'eslide)
Something went wrong with that request. Please try again.