;;; sx-question-print.el --- populating the question-mode buffer with content  -*- 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:


;;; Code:
(require 'markdown-mode)
(require 'sx-button)
(require 'sx)
(require 'sx-question)
(require 'sx-babel)
(require 'sx-user)

(defvar sx-question-mode--data)

(defgroup sx-question-mode nil
  "Customization group for sx-question-mode."
  :prefix "sx-question-mode-"
  :tag "SX Question Mode"
  :group 'sx)

(defgroup sx-question-mode-faces '((sx-user custom-group))
  "Customization group for the faces of `sx-question-mode'.
Some faces of this mode might be defined in the `sx-user' group."
  :prefix "sx-question-mode-"
  :tag "SX Question Mode Faces"
  :group 'sx-question-mode)


;;; Faces and Variables
(defface sx-question-mode-header
  '((t :inherit font-lock-variable-name-face))
  "Face used on the question headers in the question buffer."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-title
  '((t :weight bold :inherit default))
  "Face used on the question title in the question buffer."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-title-comments
  '((t :inherit sx-question-mode-title))
  "Face used on the question title in the question buffer."
  :group 'sx-question-mode-faces)

(defcustom sx-question-mode-header-title "\n"
  "String used before the question title at the header."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-header-author-format "\nAuthor:    %d %r"
  "String used to display the question author at the header.
% constructs have special meaning here.  See `sx-user--format'."
  :type 'string
  :group 'sx-question-mode)

(defface sx-question-mode-date
  '((t :inherit font-lock-string-face))
  "Face used on the question date in the question buffer."
  :group 'sx-question-mode-faces)

(defcustom sx-question-mode-header-date "\nPosted on: "
  "String used before the question date at the header."
  :type 'string
  :group 'sx-question-mode)

(defface sx-question-mode-score
  '((t))
  "Face used for the score in the question buffer."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-score-downvoted
  '((t :inherit (font-lock-warning-face sx-question-mode-score)))
  "Face used for downvoted score in the question buffer."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-score-upvoted
  '((t :weight bold
       :inherit (font-lock-function-name-face sx-question-mode-score)))
  "Face used for downvoted score in the question buffer."
  :group 'sx-question-mode-faces)

(defcustom sx-question-mode-header-tags "\nTags:      "
  "String used before the question tags at the header."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-header-score "\nScore:     "
  "String used before the question score at the header."
  :type 'string
  :group 'sx-question-mode)

(defface sx-question-mode-content-face
  '((((background dark)) :background "#090909")
    (((background light)) :background "#f4f4f4"))
  "Face used on the question body in the question buffer.
This shouldn't have a foreground, or this will interfere with
font-locking."
  :group 'sx-question-mode-faces)

(defcustom sx-question-mode-last-edit-format " (edited %s ago by %s)"
  "Format used to describe last edit date in the header.
First \"%s\" is replaced with the date and the second \"%s\" with
the editor's name."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-separator
  (concat (propertize (make-string 72 ?\s)
                      'face '(underline sx-question-mode-header))
          "\n")
  "Separator used between header and body."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-answer-title "Answer"
  "Title used at the start of \"Answer\" sections."
  :type 'string
  :group 'sx-question-mode)

(defface sx-question-mode-accepted
  '((((background dark)) :foreground "LimeGreen"
     :height 1.3 :inherit sx-question-mode-title)
    (((background light)) :foreground "ForestGreen"
     :height 1.3 :inherit sx-question-mode-title))
  "Face used for accepted answers in the question buffer."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-closed
  '((t :box 2 :inherit font-lock-warning-face))
  "Face used for closed question header in the question buffer."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-closed-reason
  `((t :box (:line-width 2 :color ,(face-attribute 'sx-question-mode-closed
                                                   :foreground nil t))
       :inherit sx-question-mode-title))
  "Face used for closed question header in the question buffer.
Aesthetically, it's important that the color of this face's :box
attribute match the color of the face `sx-question-mode-closed'."
  :group 'sx-question-mode-faces)

(defcustom sx-question-mode-answer-accepted-title "Accepted Answer"
  "Title used at the start of accepted \"Answer\" section."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-comments-title " Comments"
  "Title used at the start of \"Comments\" sections."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-comments-format "%s: %s\n"
  "Format used to display comments.
First \"%s\" is replaced with user name.  Second \"%s\" is
replaced with the comment."
  :type 'string
  :group 'sx-question-mode)

(defcustom sx-question-mode-pretty-links t
  "If non-nil, markdown links are displayed in a compact form."
  :type 'boolean
  :group 'sx-question-mode)

(defconst sx-question-mode--sort-methods
  (let ((methods
         '(("Higher-scoring" . sx-answer-higher-score-p)
           ("Newer"          . sx-answer-newer-p)
           ("More active"    . sx-answer-more-active-p))))
    (append (mapcar (lambda (x) (cons (concat (car x) " first") (cdr x)))
                    methods)
            (mapcar (lambda (x) (cons (concat (car x) " last")
                                      (sx--invert-predicate (cdr x))))
                    methods))))

(defcustom sx-question-mode-answer-sort-function
  #'sx-answer-higher-score-p
  "Function used to sort answers in the question buffer."
  :type
  (cons 'choice
        (mapcar (lambda (x) `(const :tag ,(car x) ,(cdr x)))
                sx-question-mode--sort-methods))
  :group 'sx-question-mode)

(defcustom sx-question-mode-use-images (image-type-available-p 'imagemagick)
  "Non-nil if SX should download and display images.
By default, this is `t' if the `imagemagick' image type is
available (checked with `image-type-available-p').  If this image
type is not available, images won't work."
  :type 'boolean
  :group 'sx-question-mode)

(defcustom sx-question-mode-image-max-width 550
  "Maximum width, in pixels, of images in the question buffer."
  :type 'integer
  :group 'sx-question-mode)


;;; Functions
;;;; Printing the general structure
(defconst sx-question-mode--closed-mode-line-string
  '(:propertize "  [CLOSED]  " face font-lock-warning-face)
  "String indicating closed questions in the mode-line.")

(defun sx-question-mode--print-question (question)
  "Print a buffer describing QUESTION.
QUESTION must be a data structure returned by `json-read'."
  (when (sx--deleted-p question)
    (sx-user-error "This is a deleted question"))
  (setq sx-question-mode--data question)
  ;; Clear the overlays
  (mapc #'delete-overlay sx--overlays)
  (setq sx--overlays nil)
  ;; Print everything
  (sx-assoc-let question
    (when .closed_reason
      (add-to-list 'mode-line-format sx-question-mode--closed-mode-line-string)
      (sx-question-mode--print-close-reason .closed_reason .closed_date .closed_details))
    (sx-question-mode--print-section question)
    (mapc #'sx-question-mode--print-section
          (cl-remove-if
           #'sx--deleted-p
           (cl-sort .answers sx-question-mode-answer-sort-function))))
  (insert "\n\n                       ")
  (insert-text-button "Write an Answer" :type 'sx-button-answer)
  ;; Go up
  (goto-char (point-min))
  (sx-question-mode-next-section))

(defun sx-question-mode--print-close-reason (reason date &optional details)
  "Print a header explaining REASON and DATE.
DATE is an integer.

DETAILS, when given is an alist further describing the close."
  (let ((l (point)))
    (let-alist details
      (insert "\n    "
              (propertize (format " %s as %s, %s ago. "
                                  (if .on_hold "Put on hold" "Closed")
                                  reason
                                  (sx-time-since date))
                          'face 'sx-question-mode-closed)
              "\n")
      (when .description
        (insert (replace-regexp-in-string "<[^>]+>" "" .description)
                "\n")))
    (save-excursion
      (goto-char l)
      (search-forward " as " nil 'noerror)
      (setq l (point))
      (skip-chars-forward "^,")
      (let ((ov (make-overlay l (point))))
        (overlay-put ov 'face 'sx-question-mode-closed-reason)
        (push ov sx--overlays)))))

(defun sx-question-mode--maybe-print-accept-button ()
  "Print accept button if you own this question."
  (when (sx-assoc-let sx-question-mode--data
          (ignore-errors
            (= .owner.user_id
               (cdr (assq 'user_id (sx-network-user .site_par))))))
    (insert "     ")
    (insert-text-button "Accept" :type 'sx-button-accept)))

(defun sx-question-mode--print-section (data)
  "Print a section corresponding to DATA.
DATA can represent a question or an answer."
  ;; This makes `data' accessible through `sx--data-here'.
  (sx--wrap-in-overlay
      (list 'sx--data-here data)
    (sx-assoc-let data
      (insert sx-question-mode-header-title)
      (insert-text-button
       ;; Questions have title, Answers don't
       (cond (.title)
             (.is_accepted sx-question-mode-answer-accepted-title)
             (t sx-question-mode-answer-title))
       ;; Section level
       'sx-question-mode--section (if .title 1 2)
       'sx-button-copy .share_link
       'face (if .is_accepted 'sx-question-mode-accepted
               'sx-question-mode-title)
       :type 'sx-question-mode-title)
      (when (not (or .title .is_accepted))
        (sx-question-mode--maybe-print-accept-button))

      ;; Sections can be hidden with overlays
      (sx--wrap-in-overlay
          '(sx-question-mode--section-content t)

        ;; Author
        (insert
         (sx-user--format
          (propertize sx-question-mode-header-author-format
                      'face 'sx-question-mode-header)
          .owner))

        ;; Date
        (sx-question-mode--insert-header
         sx-question-mode-header-date
         (concat
          (sx-time-seconds-to-date .creation_date)
          (when .last_edit_date
            (format sx-question-mode-last-edit-format
                    (sx-time-since .last_edit_date)
                    (sx-user--format "%d" .last_editor))))
         'sx-question-mode-date)

        ;; Score and upvoted/downvoted status.
        (sx-question-mode--insert-header
         sx-question-mode-header-score
         (format "%s%s" .score
                 (cond (.upvoted "↑") (.downvoted "↓") (t "")))
         (cond (.upvoted 'sx-question-mode-score-upvoted)
               (.downvoted 'sx-question-mode-score-downvoted)
               (t 'sx-question-mode-score)))

        ;; Tags
        (when .title
          ;; Tags
          (sx-question-mode--insert-header
           sx-question-mode-header-tags
           (sx-tag--format-tags .tags .site_par)
           nil))
        ;; Body
        (insert "\n" sx-question-mode-separator)
        (sx--wrap-in-overlay
            '(face sx-question-mode-content-face)
          (insert "\n")
          (sx-question-mode--insert-markdown .body_markdown)
          (insert "\n" sx-question-mode-separator))
        ;; Clean up commments manually deleted.  The `append' call is
        ;; to ensure `comments' is a list and not a vector.
        (let ((comments (cl-remove-if #'sx--deleted-p .comments)))
          (when comments
            (insert "\n")
            (insert-text-button
             sx-question-mode-comments-title
             'face 'sx-question-mode-title-comments
             'sx-question-mode--section 3
             'sx-button-copy .share_link
             :type 'sx-question-mode-title)
            (sx--wrap-in-overlay
                '(sx-question-mode--section-content t)
              (insert "\n")
              (sx--wrap-in-overlay
                  '(face sx-question-mode-content-face)
                ;; Comments have their own `sx--data-here' property (so they can
                ;; be upvoted too).
                (mapc #'sx-question-mode--print-comment comments))
              ;; If there are comments, we want part of this margin to go
              ;; inside them, so the button get's placed beside the
              ;; "Comments" header when you hide them.
              (insert "         ")))
          ;; If there are no comments, we have to add this margin here.
          (unless comments
            (insert "         ")))
        (insert "               ")
        ;; This is where the "add a comment" button is printed.
        (insert-text-button "Add a Comment"
                            :type 'sx-button-comment)
        (insert "\n")))))

(defun sx-question-mode--print-comment (comment-data)
  "Print the comment described by alist COMMENT-DATA.
The comment is indented, filled, and then printed according to
`sx-question-mode-comments-format'."
  (sx--wrap-in-overlay
      (list 'sx--data-here comment-data)
    (sx-assoc-let comment-data
      (when (and (numberp .score) (> .score 0))
        (insert (number-to-string .score)
                (if .upvoted "^" "")
                " "))
      (insert
       (format sx-question-mode-comments-format
               (sx-user--format "%d" .owner)
               (substring
                ;; We use temp buffer, so that image overlays don't get
                ;; inserted with the comment.
                (with-temp-buffer
                  ;; We fill with three spaces at the start, so the comment is
                  ;; slightly indented.
                  (sx-question-mode--insert-markdown (concat "   " (sx--squash-whitespace .body_markdown)))
                  (buffer-string))
                ;; Then we remove the spaces from the first line, since we'll
                ;; add the username there anyway.
                3))))))

(defun sx-question-mode--insert-header (&rest args)
  "Insert propertized ARGS.
ARGS is a list of repeating values -- `header', `value', and
`face'.  `header' is given `sx-question-mode-header' as a face,
where `value' is given `face' as its face.

\(fn HEADER VALUE FACE [HEADER VALUE FACE] [HEADER VALUE FACE] ...)"
  (while args
    (insert
     (propertize (pop args) 'face 'sx-question-mode-header)
     (let ((header (pop args))
           (face (pop args)))
       (if face (propertize header 'face face)
         header)))))


;;;; Printing and Font-locking the content (body)
(defvar sx-question-mode-bullet-appearance
  (propertize (if (char-displayable-p ?•) "•" "*")
              'face 'markdown-list-face)
  "String to be displayed as the bullet of markdown list items.")

(defconst sx-question-mode--reference-regexp
  (rx line-start (0+ blank) "[%s]:" (0+ blank)
      (group-n 1 (1+ (not (any blank "\n\r")))))
  "Regexp used to find the url of labeled links.
E.g.:
   [1]: https://...")

(defconst sx-question-mode--link-regexp
  ;; Done at compile time.
  (rx (or (and "[" (optional (group-n 6 "meta-")) "tag:"
               (group-n 5 (+ (not (any " ]")))) "]")
          (and (opt "!") "[" (group-n 1 (1+ (not (any "[]")))) "]"
               (or (and "(" (group-n 2 (1+ (not (any ")")))) ")")
                   (and "[" (group-n 3 (1+ (not (any "]")))) "]")))
          (group-n 4 (and "http" (opt "s") "://"
                          (>= 2 (any lower numeric "_%"))
                          "."
                          (>= 2 (any lower numeric "_%"))
                          (* (any lower numeric "-/._%&#?=;"))))))
  "Regexp matching markdown links.")

(defun sx-question-mode--process-line-breaks (beg end-marker)
  "Process Markdown line breaks between BEG and END-MARKER.
Double space at the end of a line becomes an invisible \"\\n\".
Consecutive blank lines beyond the first are consensed.
Assumes `marker-insertion-type' of END-MARKER is t."
  (goto-char beg)
  (while (search-forward-regexp
          (rx line-start (* blank) "\n"
              (group-n 1 (+ (any blank "\n"))))
          end-marker 'noerror)
    ;; An invisible newline ensures the previous text
    ;; will get filled as a separate paragraph.
    (replace-match "" nil nil nil 1))
  (goto-char beg)
  (while (search-forward-regexp "  $" end-marker 'noerror)
    ;; An invisible newline ensures the previous text
    ;; will get filled as a separate paragraph.
    (replace-match (propertize "\n" 'invisible t))))

(defun sx-question-mode--process-markdown-in-region (beg end)
  "Process Markdown text between BEG and END.
This does not do Markdown font-locking.  Instead, it fills text,
propertizes links, inserts images, cleans up html comments, and
font-locks code-blocks according to mode."
  ;; Paragraph filling
  (let ((paragraph-start
         "\f\\|[ \t]*$\\|[ \t]*[*+-] \\|[ \t]*[0-9]+\\.[ \t]\\|[ \t]*: ")
        (paragraph-separate "\\(?:[ \t\f]*\\|.*  \\)$")
        (adaptive-fill-first-line-regexp "\\`[ \t]*>[ \t]*?\\'")
        (adaptive-fill-function #'markdown-adaptive-fill-function)) 
    (save-restriction
      (narrow-to-region beg end)
      ;; html tags can span many paragraphs, so we handle them
      ;; globally first.
      (sx-question-mode--process-html-tags (point-min) (copy-marker (point-max)))
      ;; And now the filling and other handlings.
      (goto-char (point-min))
      (while (null (eobp))
        ;; Don't fill pre blocks.
        (unless (sx-question-mode--dont-fill-here)
          (let ((beg (point)))
            (skip-chars-forward "\r\n[:blank:]")
            (forward-paragraph)
            (let ((end (point-marker)))
              (set-marker-insertion-type end t)
              ;; Turn markdown linebreaks into their final form
              (sx-question-mode--process-line-breaks beg end)
              ;; Compactify links by paragraph, so we don't linkify
              ;; inside code-blocks. This will still linkify inside
              ;; code tags, unfortunately.
              (sx-question-mode--process-links beg end)
              ;; Filling is done after all of the above, since those
              ;; steps change the length of text.
              (fill-region beg end)
              (goto-char end)))))
      (goto-char (point-max)))))

(defconst sx-question-mode-hr
  (propertize (make-string 72 ?―)
              'face 'markdown-header-rule-face))

(defun sx-question-mode--insert-markdown (text)
  "Return TEXT fontified according to `markdown-mode'."
  (let ((beg (point)))
    (insert
     ;; Font-locking needs to be done in a temp buffer, because it
     ;; affects the entire buffer even if we narrow.
     (with-temp-buffer
       (insert text)
       ;; Trim whitespace
       (goto-char (point-max))
       (skip-chars-backward "\r\n[:blank:]")
       (delete-region (point) (point-max))
       (goto-char (point-min))
       (skip-chars-forward "\r\n[:blank:]")
       (forward-line 0)
       (delete-region (point-min) (point))
       ;; Font lock
       (delay-mode-hooks (markdown-mode))
       (font-lock-mode -1)
       (when sx-question-mode-bullet-appearance
         (font-lock-add-keywords ;; Bullet items.
          nil
          `((,(rx line-start (0+ blank) (group-n 1 (any "*+-")) blank)
             1 '(face nil display ,sx-question-mode-bullet-appearance) prepend))))
       (font-lock-add-keywords ;; Highlight usernames.
        nil
        `((,(rx (or blank line-start)
                (group-n 1 (and "@" (1+ (not space))))
                symbol-end)
           1 font-lock-builtin-face)
          ("^---+$" 0 '(face nil display ,sx-question-mode-hr))))
       ;; Everything.
       (font-lock-fontify-region (point-min) (point-max))
       (replace-regexp-in-string "[[:blank:]]+\\'" "" (buffer-string))))
    ;; This part can and should be done in place, this way it can
    ;; create overlays.
    (sx-question-mode--process-markdown-in-region beg (point))))


;;; HTML tags
(defconst sx-question-mode--html-tag-regexp
  (rx "<" (group-n 1 "%s") (* (not (any ">"))) ">"))

(defface sx-question-mode-sub-sup-tag
  '((t :height 0.7))
  "Face used on <sub> and <sup> tags."
  :group 'sx-question-mode-faces)

(defface sx-question-mode-kbd-tag
  '((((background dark))
     :height 0.9
     :weight semi-bold
     :box (:line-width 3 :style released-button :color "gray30"))
    (((background light))
     :height 0.9
     :weight semi-bold
     :box (:line-width 3 :style released-button :color "gray70")))
  "Face used on <kbd> tags."
  :group 'sx-question-mode-faces)

(defun sx-question-mode--inside-code-p ()
  "Return non-nil if point is inside code.
This can be inline Markdown code or a Markdown code-block."
  (save-match-data
    (or (markdown-code-at-point-p)
        (save-excursion
          (sx-question-mode--skip-and-fontify-pre 'dont-fontify)))))

(defun sx-question-mode--standalone-tag-p (string)
  "Return non-nil if STRING ends in \"/>\"."
  (string-match "/[[:blank:]]*>\\'" string))

(defun sx-question-mode--next-tag (tag &optional closing end)
  "Move point to the next occurrence of html TAG, or return nil.
Don't move past END.
If CLOSING is non-nil, find a closing tag."
  (search-forward-regexp
   (format sx-question-mode--html-tag-regexp
           (if closing
               (concat "/[[:blank:]]*" tag)
             tag))
   end 'noerror))

(defun sx-question-mode--process-html-tags (beg end-marker)
  "Hide all html tags between BEG and END and possibly interpret them.
END-MARKER should be a marker."
  ;; This code understands nested html, but not if the same tag is
  ;; nested in itself (e.g., <kbd><kbd></kbd></kbd>).
  (set-marker-insertion-type end-marker t)
  (goto-char beg)
  (while (sx-question-mode--next-tag "[[:alpha:]]+" nil end-marker)
    (unless (sx-question-mode--inside-code-p)
      (let ((tag (match-string 1))
            (full (match-string 0))
            (l   (match-beginning 0)))
        (replace-match "")
        (pcase tag
          (`"hr"
           (unless (looking-at-p "^") (insert "\n"))
           (insert (propertize "---" 'display sx-question-mode-hr))
           (unless (eq (char-after) ?\n) (insert "\n")))
          (`"br" (insert "\n  ")))
        (when (and (not (sx-question-mode--standalone-tag-p full))
                   (sx-question-mode--next-tag tag 'closing))
          (let ((r (copy-marker (match-beginning 0))))
            ;; The code tag is special, because it quotes everything inside.
            (if (string= tag "code")
                (progn (replace-match "`")
                       (save-excursion (goto-char l) (insert "`")))
              (replace-match "")
              ;; Handle stuff between the two tags.
              (save-match-data (sx-question-mode--process-html-tags l r))
              (pcase tag
                (`"kbd"
                 (add-text-properties l r '(face sx-question-mode-kbd-tag))
                 (when (looking-at-p
                        (format sx-question-mode--html-tag-regexp "kbd"))
                   (insert " ")))
                (`"sub"
                 (add-text-properties
                  l r '(face sx-question-mode-sub-sup-tag display (raise -0.3))))
                (`"sup"
                 (add-text-properties
                  l r '(face sx-question-mode-sub-sup-tag display (raise +0.3))))))))))))


;;; Handling links
(defun sx-question-mode--process-links (beg end-marker)
  "Turn all markdown links between BEG and ENG into compact format.
Image links are downloaded and displayed, if
`sx-question-mode-use-images' is non-nil.
Assumes `marker-insertion-type' of END-MARKER is t."
  (goto-char beg)
  (while (search-forward-regexp sx-question-mode--link-regexp end-marker t)
    ;; Tags are tag-buttons.
    (let ((tag (match-string-no-properties 5)))
      (if (and tag (> (length tag) 0))
          (progn (replace-match "")
                 (sx-tag--insert tag))
        ;; Other links are link-buttons.
        (let* ((text (match-string-no-properties 1))
               (url (or (match-string-no-properties 2)
                        (match-string-no-properties 4)
                        (sx-question-mode-find-reference
                         (match-string-no-properties 3)
                         text)))
               (full-text (match-string-no-properties 0))
               (image-p (and sx-question-mode-use-images
                             (eq ?! (elt full-text 0)))))
          (when (stringp url)
            (replace-match "")
            (sx-question-mode--insert-link
             (cond (image-p (sx-question-mode--create-image url))
                   ((and sx-question-mode-pretty-links text))
                   ((not text) (sx--shorten-url url))
                   (t full-text))
             url)))))))

(defun sx-question-mode--create-image (url)
  "Get and create an image from URL and insert it at POINT.
The image will take the place of the character at POINT.
Its size is bound by `sx-question-mode-image-max-width' and
`window-body-width'."
  (let* ((ov (make-overlay (point) (point) (current-buffer) t nil))
         (callback
          (lambda (data)
            (let* ((image (create-image data 'imagemagick t))
                   (image-width (car (image-size image 'pixels))))
              (overlay-put
               ov 'display
               (append image
                       (list :width (min sx-question-mode-image-max-width
                                         (window-body-width nil 'pixel)
                                         image-width))))))))
    (sx-request-get-url url callback)
    (overlay-put ov 'face 'default)
    ov))

(defun sx-question-mode--insert-link (text url)
  "Return a link propertized version of TEXT-OR-IMAGE.
URL is used as 'help-echo and 'url properties."
  ;; Try to handle an image/link inside another link.
  (when (eq (char-before) ?\[)
    (insert "a")
    (forward-char -2)
    (if (looking-at sx-question-mode--link-regexp)
        (progn (setq url (or (match-string-no-properties 2)
                             (match-string-no-properties 4)
                             (sx-question-mode-find-reference
                              (match-string-no-properties 3)
                              (if (stringp text) text "¶"))
                             url))
               (replace-match ""))
      (forward-char 1)
      (delete-char 1)))
  (unless (stringp text)
    ;; Images need to be at the start of a line.
    (unless (looking-at-p "^") (insert "\n"))
    ;; And need an empty line above so they don't get wrapped into
    ;; text when we do filling.
    (insert (propertize "\n" 'display "")))
  ;; Insert the link button.
  (insert-text-button (if (stringp text) text "¶")
                      ;; Mouse-over
                      'help-echo
                      (format sx-button--link-help-echo
                              ;; If TEXT is a shortened url, we don't shorten URL.
                              (propertize (if (and (stringp text)
                                                   (string-match "^https?:" text))
                                              url (sx--shorten-url url))
                                          'face 'font-lock-function-name-face))
                      ;; For visiting and stuff.
                      'sx-button-url url
                      'sx-button-copy url
                      :type 'sx-button-link)
  ;; Images need to be at the end of a line too.
  (unless (stringp text)
    (move-overlay text (1- (point)) (point) (current-buffer))
    (insert (propertize "\n\n" 'display "\n"))))

(defun sx-question-mode-find-reference (id &optional fallback-id)
  "Find url identified by reference ID in current buffer.
If ID is nil, use FALLBACK-ID instead."
  (save-excursion
    (save-match-data
      (goto-char (point-min))
      (when (search-forward-regexp
             (format sx-question-mode--reference-regexp
                     (or id fallback-id))
             nil t)
        (match-string-no-properties 1)))))


;;; Things we don't fill
(defun sx-question-mode--dont-fill-here ()
  "If text shouldn't be filled here, return t and skip over it."
  (catch 'sx-question-mode-done
    (let ((before (point)))
      (skip-chars-forward "\r\n[:blank:]")
      (let ((first-non-blank (point)))
        (dolist (it '(sx-question-mode--skip-and-fontify-pre
                      sx-question-mode--skip-headline
                      sx-question-mode--skip-references
                      sx-question-mode--skip-comments))
          ;; If something worked, keep point where it is and return t.
          (if (funcall it) (throw 'sx-question-mode-done t)
            ;; Before calling each new function. Go back to the first
            ;; non-blank char.
            (goto-char first-non-blank)))
        ;; If nothing matched, go back to the very beginning.
        (goto-char before)
        ;; And return nil
        nil))))

(defun sx-question-mode--skip-and-fontify-pre (&optional dont-fontify)
  "If there's a pre block ahead, handle it, skip it and return t.
Handling means to turn it into a button and remove erroneous
font-locking.

If DONT-FONTIFY is non-nil, just return the result and possibly
move point, don't create the code-block button."
  (let ((beg (line-beginning-position)))
    ;; To identify code-blocks we need to be at start of line.
    (goto-char beg)
    (when (fboundp 'markdown-syntax-propertize)
      (markdown-syntax-propertize (point) (point-max)))
    (when (markdown-match-pre-blocks (line-end-position))
      (unless dont-fontify
        (sx-babel--make-pre-button beg (point)))
      t)))

(defun sx-question-mode--skip-comments ()
  "If there's an html comment ahead, skip it and return t."
  ;; @TODO: Handle the comment.
  ;; "Handling means to store any relevant metadata it might be holding."
  (let ((end (save-excursion
               (when (markdown-match-comments (line-end-position))
                 (point)))))
    (when end
      (delete-region (point) end)
      (skip-chars-backward "[:blank:]")
      (when (looking-at "^[:blank:]*\n")
        (replace-match ""))
      t)))

(defun sx-question-mode--skip-headline ()
  "If there's a headline ahead, skip it and return non-nil."
  (when (or (looking-at-p "^#+ ")
            (progn (forward-line 1) (looking-at-p "===\\|---")))
    ;; Returns non-nil.
    (forward-line 1)))

(defun sx-question-mode--skip-references ()
  "If there's a reference ahead, skip it and return non-nil."
  (forward-line 0)
  (when (looking-at-p (format sx-question-mode--reference-regexp ".+"))
    ;; Returns non-nil
    (forward-paragraph 1)
    t))

(provide 'sx-question-print)
;;; sx-question-print.el ends here

;; Local Variables:
;; indent-tabs-mode: nil
;; End: