216 lines
7.1 KiB
EmacsLisp
Raw Normal View History

2016-09-22 15:58:26 +00:00
;;; sx-button.el --- defining buttons -*- lexical-binding: t; -*-
;; Copyright (C) 2014 Artur Malabarba
;; Author: Artur Malabarba <bruce.connor.am@gmail.com>
;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.
;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with this program. If not, see <http://www.gnu.org/licenses/>.
;;; Commentary:
;;
;; This file defines all buttons used by SX. For information on
;; buttons, see:
;; http://www.gnu.org/software/emacs/manual/html_node/elisp/Buttons.html
;;
;; Most interactive parts of the SX buffers are buttons. Wherever you
;; are, you can always cycle through all buttons by hitting `TAB',
;; that should help identify what's a button in each buffer.
;;
;; To define a new type of button follow the examples below using
;; `define-button-type' with :supertype `sx-button'. Required
;; properties are `action' and `help-echo'. You'll probably want to
;; give it a `face' as well, unless you want it to look like a link.
;;
;; Buttons can then be inserted in their respective files using
;; `insert-text-button'. Give it the string, the `:type' you defined,
;; and any additional properties that can only be determined at
;; creation. Existing text can be transformed into a button with
;; `make-text-button' instead.
;;; Code:
(require 'button)
(require 'sx)
(require 'sx-question)
(declare-function sx-accept "sx-interaction")
(declare-function sx-answer "sx-interaction")
(declare-function sx-comment "sx-interaction")
(declare-function sx-open-link "sx-interaction")
(declare-function sx-question-mode-hide-show-section "sx-question-mode")
;;; Face
(defface sx-custom-button
'((((type x w32 ns) (class color)) ; Like default mode line
:box (:line-width 3 :style released-button)
:height 0.9
:background "lightgrey" :foreground "black"))
"Face used on buttons such as \"Write an Answer\"."
:group 'sx)
;;; Command definitions
;; This extends `button-map', which already defines RET and mouse-1.
(defvar sx-button-map
(let ((map (copy-keymap button-map)))
(define-key map "w" #'sx-button-copy)
map)
"Keymap used on buttons.")
(defun sx-button-copy ()
"Copy the content of thing at point.
This is usually a link's URL, or the content of a code block."
(interactive)
(let ((content
(get-text-property (point) 'sx-button-copy)))
(if (null content)
(sx-message "Nothing to copy here.")
(kill-new content)
(sx-message "Copied %s to kill ring."
(or (get-text-property
(point) 'sx-button-copy-type)
content)))))
(defun sx-button-edit-this (text-or-marker &optional majormode)
"Open a temp buffer populated with the string TEXT-OR-MARKER using MAJORMODE.
When given a marker (or interactively), use the 'sx-button-copy
and the 'sx-mode text-properties under the marker. These are
usually part of a code-block."
(interactive (list (point-marker)))
;; Buttons receive markers.
(when (markerp text-or-marker)
(setq majormode (get-text-property text-or-marker 'sx-mode))
(unless (setq text-or-marker
(get-text-property text-or-marker 'sx-button-copy))
(sx-message "Nothing of interest here.")))
(with-current-buffer (pop-to-buffer (generate-new-buffer
"*sx temp buffer*"))
(insert text-or-marker)
(when majormode
(funcall majormode))))
(defun sx-button-follow-link (&optional pos)
"Follow link at POS. If POS is nil, use `point'."
(interactive)
(let ((url (or (get-text-property (or pos (point)) 'sx-button-url)
(sx-user-error "No url under point: %s" (or pos (point))))))
;; If we didn't recognize the link, this errors immediately. If
;; we mistakenly recognize it, it will error when we try to fetch
;; whatever we thought it was.
(condition-case nil (sx-open-link url)
;; When it errors, don't blame the user, just visit externally.
(error (browse-url url)))))
;;; Help-echo definitions
(defconst sx-button--help-echo
(concat "mouse-1, RET"
(propertize ": %s -- " 'face 'minibuffer-prompt)
"w"
(propertize ": copy %s" 'face 'minibuffer-prompt))
"Base help-echo on which others can be written.")
(defconst sx-button--user-help-echo
(format sx-button--help-echo
"visit user page"
"link")
"Help echoed in the minibuffer when point is on a user.")
(defconst sx-button--tag-help-echo
(format sx-button--help-echo
"Tag search"
"tag")
"Help echoed in the minibuffer when point is on a tag.")
(defconst sx-button--question-title-help-echo
(format sx-button--help-echo
"hide content"
"link")
"Help echoed in the minibuffer when point is on a section.")
(defconst sx-button--link-help-echo
(format sx-button--help-echo
"visit %s"
"URL")
"Help echoed in the minibuffer when point is on a section.")
;;; Type definitions
(define-button-type 'sx-button
'follow-link t
'keymap sx-button-map)
(define-button-type 'sx-question-mode-title
'face 'sx-question-mode-title
'action #'sx-question-mode-hide-show-section
'help-echo sx-button--question-title-help-echo
'sx-button-copy-type "Share Link"
:supertype 'sx-button)
(define-button-type 'sx-question-mode-code-block
'action #'sx-button-edit-this
'face nil
:supertype 'sx-button)
(define-button-type 'sx-button-link
'action #'sx-button-follow-link
:supertype 'sx-button)
(define-button-type 'sx-button-user
'action #'sx-button-follow-link
'help-echo sx-button--user-help-echo
;; We use different faces on different parts of the user button.
'face 'sx-user-name
:supertype 'sx-button)
(declare-function sx-search-tag-at-point "sx-search")
(define-button-type 'sx-button-tag
'action #'sx-search-tag-at-point
'help-echo sx-button--tag-help-echo
'face 'sx-tag
:supertype 'sx-button)
(define-button-type 'sx-button-comment
'help-echo (concat "mouse-1, RET"
(propertize ": write a comment"
'face 'minibuffer-prompt))
'face 'sx-custom-button
'action #'sx-comment
:supertype 'sx-button)
(define-button-type 'sx-button-accept
'help-echo (concat "mouse-1, RET"
(propertize ": accept answer"
'face 'minibuffer-prompt))
'face 'sx-custom-button
'action #'sx-accept
:supertype 'sx-button)
(define-button-type 'sx-button-answer
'help-echo (concat "mouse-1, RET"
(propertize ": write an answer"
'face 'minibuffer-prompt))
'face 'sx-custom-button
'action #'sx-answer
:supertype 'sx-button)
(provide 'sx-button)
;;; sx-button.el ends here
;; Local Variables:
;; indent-tabs-mode: nil
;; End: