diff --git a/.gitignore b/.gitignore
index c146d02c..fec4707f 100644
--- a/.gitignore
+++ b/.gitignore
@@ -13,6 +13,7 @@ example/tags/
example/2*
example/A-Non-Post-Scribble-Page*
example/img/posts/
+example/img/resized/
# Emacs / DrRacket backups & auto save files
*~
diff --git a/example/.frogrc b/example/.frogrc
index 98a12f77..cea22570 100644
--- a/example/.frogrc
+++ b/example/.frogrc
@@ -1,4 +1,6 @@
-# Required: Should NOT end in trailing slash.
+# -*- conf -*-
+
+# Required: Should NOT end in trailing slash.
scheme/host = http://www.example.com
# A path prepended to URIs, including those specified here in .frogrc
@@ -117,3 +119,20 @@ python-executable = python
pygments-linenos? = true
## CSS class for the wrapping
tag (default: 'highlight').
pygments-cssclass = source
+
+# Serve responsive images.
+#
+# Make use of the img srcset attribute to serve images inside elements
+# of class "figure" (such as image referenced from Markdown) at three
+# different sizes. Depends on having ImageMagick installed.
+responsive-images? = true
+
+# Subdirectory of where to put resized images. Defaults to "resized".
+# The directory will be created but the parent "img" directory must exist.
+#image-output-dir = resized
+
+# Value of the img "sizes" attribute.
+# Defaults to "(max-width: px) 100vw, px"
+# If your blog's main column is narrower than the page width on wide
+# clients you may want to have something like:
+#image-sizes-attr = (max-width: ) 100vw,
diff --git a/example/_src/posts/2013-06-19-a-scribble-post.scrbl b/example/_src/posts/2013-06-19-a-scribble-post.scrbl
index 43ef2eff..0b4dab97 100644
--- a/example/_src/posts/2013-06-19-a-scribble-post.scrbl
+++ b/example/_src/posts/2013-06-19-a-scribble-post.scrbl
@@ -110,3 +110,9 @@ function foo() {
return 7;
}
}
+
+@subsection[#:style 'unnumbered]{B SubSection}
+
+A responsive big black image:
+
+@image["img/800px-image.gif" #:style "img-responsive"]
diff --git a/example/_src/posts/2016-08-12-a-blog-post-featuring-a-big-image.md b/example/_src/posts/2016-08-12-a-blog-post-featuring-a-big-image.md
new file mode 100644
index 00000000..522c446c
--- /dev/null
+++ b/example/_src/posts/2016-08-12-a-blog-post-featuring-a-big-image.md
@@ -0,0 +1,7 @@
+ Title: A blog post featuring a big image
+ Date: 2016-08-12T02:43:56
+ Tags: images, responsive
+
+The below `img` tag should come with _srcset_ and _sizes_ definitions:
+
+![Image title](/img/800px-image.gif)
diff --git a/example/img/1300px-image.gif b/example/img/1300px-image.gif
new file mode 100644
index 00000000..0be15ae6
Binary files /dev/null and b/example/img/1300px-image.gif differ
diff --git a/example/img/600px-image.gif b/example/img/600px-image.gif
new file mode 100644
index 00000000..4666a934
Binary files /dev/null and b/example/img/600px-image.gif differ
diff --git a/example/img/800px-image.gif b/example/img/800px-image.gif
new file mode 100644
index 00000000..d0941388
Binary files /dev/null and b/example/img/800px-image.gif differ
diff --git a/frog/enhance-body.rkt b/frog/enhance-body.rkt
index 931ba569..14cf6325 100644
--- a/frog/enhance-body.rkt
+++ b/frog/enhance-body.rkt
@@ -13,7 +13,10 @@
"html.rkt"
"params.rkt"
"pygments.rkt"
- "xexpr-map.rkt")
+ "xexpr-map.rkt"
+ "verbosity.rkt"
+ "paths.rkt"
+ "responsive-images.rkt")
(provide enhance-body)
@@ -21,10 +24,151 @@
(define (enhance-body xs)
(~> xs
+ responsive-images
syntax-highlight
add-racket-doc-links
auto-embed-tweets))
+(define responsive-images
+ (let ([magick-notice-displayed? #f])
+ (λ (xs)
+ (define (remote-host url)
+ (url-host (string->url url)))
+ (define (do-it xs)
+ (for/list ([x xs])
+ (match x
+ [`(div ([class ,classes])
+ (img ,(list-no-order `[src ,url] attrs ...))
+ ,content ...)
+ #:when (and (regexp-match #px"\\bfigure\\b" classes)
+ (not (remote-host url)))
+ (let ([sizes-attr (assq 'sizes attrs)])
+ `(div ([class ,classes])
+ (img ([class "img-responsive"] ; Add Bootstrap class
+ ,@(make-responsive url (cond [sizes-attr => second]
+ [#t #f]))
+ ,@(if sizes-attr
+ (remove sizes-attr attrs)
+ attrs)))
+ ,@content))
+ ]
+ ;; xexpr-map?
+ [`(p () (img ,(list-no-order `[src ,url] `[class ,classes] attrs ...)))
+ #:when (and (regexp-match #px"\\bimg-responsive\\b" classes)
+ (not (remote-host url)))
+ `(p () (img ([class ,classes]
+ ,@(make-responsive url #f) ; TODO honor custom sizes?
+ ,@attrs)))]
+ [x x])))
+ (cond [(current-responsive-images?)
+ (if magick-available?
+ (do-it xs)
+ (begin
+ (unless magick-notice-displayed?
+ (prn1 "ImageMagick not found. Omitting img srcset attributes.")
+ (set! magick-notice-displayed? #t))
+ xs))]
+ [else xs]))))
+
+
+(module+ test
+ (parameterize ([top example]
+ [current-responsive-images? #t]
+ [current-image-output-dir "resized"]
+ [current-image-sizes-attr #f]
+ [current-image-sizes '(320 600 1200)]
+ [current-image-default-size 600]
+ [current-verbosity 0])
+ (test-equal? "Remote images"
+ (responsive-images
+ '((div ((class "figure")) (img ((src "//somehost.com/img/file.jpg"))))))
+ ;; Don't resize remote images. Or should we fetch it and resize it?
+ '((div ((class "figure")) (img ((src "//somehost.com/img/file.jpg"))))))
+ (when magick-available?
+ (test-equal? "Element-specific custom sizes attribute"
+ (responsive-images
+ '((div ([class "figure"])
+ (img ([src "/img/1x1.gif"]
+ [sizes "some-custom-size-spec"])))))
+ '((div ((class "figure"))
+ (img ([class "img-responsive"]
+ [src "/img/1x1.gif"]
+ [srcset "/img/1x1.gif 2w"]
+ [sizes "some-custom-size-spec"])))))
+ (test-equal? "Img with img-responsive class inside p tag"
+ (responsive-images
+ '((p () (img ([src "/img/1x1.gif"]
+ [alt ""]
+ [class "img-responsive among-others"]
+ [foo-attr "bar"])))))
+ '((p () (img ([class "img-responsive among-others"]
+ [src "/img/1x1.gif"]
+ [srcset "/img/1x1.gif 2w"]
+ [sizes "(max-width: 2px) 100vw, 2px"]
+ [alt ""]
+ [foo-attr "bar"])))))
+ (test-equal? "Image bigger than maximum size"
+ (responsive-images
+ '((div ([class "figure pull-right"])
+ (img ([src "/img/1300px-image.gif"] (alt "")))
+ (p ([class "caption"]) "some text"))))
+ `((div ((class "figure pull-right"))
+ (img ([class "img-responsive"]
+ [src "/img/resized/600/1300px-image.gif"]
+ [srcset
+ ,(string-join
+ (for/list ([s (current-image-sizes)])
+ (format "/img/resized/~a/1300px-image.gif ~aw" s s))
+ ", ")]
+ [sizes "(max-width: 1300px) 100vw, 1300px"]
+ (alt "")))
+ (p ((class "caption")) "some text"))))
+ (test-equal? "Image smaller than biggest size but bigger than smallest size"
+ (responsive-images
+ '((div ((class "figure"))
+ (img ((src "/img/800px-image.gif") (alt "")))
+ (p ((class "caption")) "some text"))))
+ `((div ((class "figure"))
+ (img ([class "img-responsive"]
+ (src ,(format "/img/resized/~a/800px-image.gif"
+ (current-image-default-size)))
+ (srcset
+ ,(string-append
+ (string-join
+ (for/list ([s '(320 600)])
+ (format "/img/resized/~a/800px-image.gif ~aw" s s))
+ ", ")
+ ", /img/800px-image.gif 800w"))
+ (sizes "(max-width: 800px) 100vw, 800px")
+ (alt "")))
+ (p ((class "caption")) "some text"))))
+ (test-equal? "Image equal to a one of the sizes specified"
+ (responsive-images
+ '((div ((class "figure"))
+ (img ((src "/img/600px-image.gif") (alt "")))
+ (p ((class "caption")) "some text"))))
+ '((div ((class "figure"))
+ (img ([class "img-responsive"]
+ (src "/img/600px-image.gif")
+ (srcset "/img/resized/320/600px-image.gif 320w, /img/600px-image.gif 600w")
+ (sizes "(max-width: 600px) 100vw, 600px")
+ (alt "")))
+ (p ((class "caption")) "some text"))))
+ (test-equal? "Image smaller than smallest size"
+ (responsive-images
+ '((div ((class "figure"))
+ (img ((src "/img/1x1.gif") (alt ""))) ; Tiny image
+ (p ((class "caption")) "some text"))))
+ '((div ((class "figure"))
+ (img ([class "img-responsive"]
+ (src "/img/1x1.gif")
+ (srcset "/img/1x1.gif 2w")
+ (sizes "(max-width: 2px) 100vw, 2px")
+ (alt "")))
+ (p ((class "caption")) "some text"))))
+ (wait-resize-images)
+ (clean-resized-images))))
+
(define (syntax-highlight xs)
(for/list ([x xs])
(match x
@@ -131,6 +275,7 @@
"&hide_thread=true"))))
(define js (call/input-url oembed-url get-pure-port read-json))
(define html ('html js))
+
(cond [html (~>> (with-input-from-string html read-html-as-xexprs)
(append '(div ([class "embed-tweet"]))))]
[else x])]
diff --git a/frog/frog.rkt b/frog/frog.rkt
index 575d6941..e8e7bbd8 100644
--- a/frog/frog.rkt
+++ b/frog/frog.rkt
@@ -25,7 +25,8 @@
"tags.rkt"
"util.rkt"
"verbosity.rkt"
- "watch-dir.rkt")
+ "watch-dir.rkt"
+ "responsive-images.rkt")
(provide serve)
(module+ test
@@ -64,7 +65,12 @@
[output-dir "."]
[python-executable "python"]
[pygments-linenos? #t]
- [pygments-cssclass "source"])
+ [pygments-cssclass "source"]
+ [responsive-images? #f]
+ [image-output-dir "resized"]
+ [image-sizes-attr #f]
+ [image-sizes '(320 768 1024)]
+ [image-default-size 768])
(define watch? #f)
(define port 3000)
(define root
@@ -312,7 +318,9 @@
(map full-uri
(append (map post-uri-path (filter linked-post?
(hash-values new-posts)))
- non-post-pages))))))
+ non-post-pages)))))
+ (when (current-responsive-images?)
+ (wait-resize-images)))
;;----------------------------------------------------------------------------
@@ -351,7 +359,8 @@
(clean-post-output-files)
(clean-non-post-output-files)
(clean-tag-output-files)
- (clean-serialized-posts))
+ (clean-serialized-posts)
+ (clean-resized-images))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
diff --git a/frog/params.rkt b/frog/params.rkt
index 8dcb0b8d..4ac63648 100644
--- a/frog/params.rkt
+++ b/frog/params.rkt
@@ -40,3 +40,8 @@
(define current-python-executable (make-parameter "python"))
(define current-pygments-linenos? (make-parameter #t))
(define current-pygments-cssclass (make-parameter "source"))
+(define current-responsive-images? (make-parameter #f))
+(define current-image-output-dir (make-parameter "resized"))
+(define current-image-sizes-attr (make-parameter #f))
+(define current-image-sizes (make-parameter '(320 768 1024)))
+(define current-image-default-size (make-parameter 768))
diff --git a/frog/responsive-images.rkt b/frog/responsive-images.rkt
new file mode 100644
index 00000000..7f9c845d
--- /dev/null
+++ b/frog/responsive-images.rkt
@@ -0,0 +1,213 @@
+#lang racket/base
+
+(require net/url
+ racket/contract/base
+ racket/contract/region
+ racket/file
+ racket/function
+ racket/list
+ racket/path
+ racket/port
+ racket/system
+ racket/string
+ (only-in racket/future processor-count)
+ (only-in racket/match match-let-values)
+ rackjure/threading
+ rackjure/str
+ "params.rkt"
+ "util.rkt"
+ "verbosity.rkt"
+ "paths.rkt")
+
+(provide make-responsive wait-resize-images clean-resized-images magick-available?)
+
+(module+ test
+ (require rackunit))
+
+(define *max-jobs* (* 1.5 (processor-count))) ; Arbitrary heuristic
+
+;; Depend on ImageMagick
+(define identify (find-executable-path "identify"))
+(define mogrify (find-executable-path "mogrify"))
+
+(define magick-available? (and identify mogrify))
+
+(define (image-width path)
+ (if (file-exists? path)
+ (~> (with-output-to-string
+ (λ ()
+ (system* identify "-format" "%w" path)))
+ string-trim
+ string->number)
+ (raise-argument-error 'image-width "Existing file" path)))
+
+(module+ test
+ (when magick-available?
+ (parameterize ([top example]
+ [current-verbosity 99])
+ (check-eq? (image-width (build-path (www/img-path) "800px-image.gif")) 800))))
+
+(struct job (input out-path width) #:transparent)
+
+(define (magick-args j)
+ ;; Imagemagick options from
+ ;; https://www.smashingmagazine.com/2015/06/efficient-image-resizing-with-
+ ;; imagemagick/
+ `("-filter" "Triangle"
+ "-define" "filter:support=2"
+ "-unsharp" "0.25x0.08+8.3+0.045"
+ "-dither" "None"
+ "-posterize" "136"
+ "-quality" "82"
+ "-define" "jpeg:fancy-upsampling=off"
+ "-define" "png:compression-filter=5"
+ "-define" "png:compression-level=9"
+ "-define" "png:compression-strategy=1"
+ "-define" "png:exclude-chunk=all"
+ "-interlace" "none"
+ "-colorspace" "sRGB"
+ "-thumbnail" ,(number->string (job-width j))
+ "-path" ,(job-out-path j)
+ ,(job-input j)))
+
+(define master-worker
+ (thread
+ (λ ()
+ (define (start-job j)
+ (match-let-values ([(proc _ _ _) (apply subprocess
+ (current-output-port)
+ (current-input-port)
+ (current-error-port)
+ mogrify (magick-args j))])
+ proc))
+ ;; N.B: Config parameters set in the main thread are reset here
+ ;; so make sure we do not rely on them. In particular prn1 and
+ ;; prn2 will not output anything.
+ (let ([finish #f]
+ [mailbox (thread-receive-evt)])
+ (let loop ([queue '()]
+ [procs '()])
+ (let ([res (apply sync mailbox procs)])
+ (cond
+ [(subprocess? res) ; Process terminated?
+ (let ([status (subprocess-status res)])
+ (unless (zero? status)
+ (eprintf "~a terminated with non-zero exit code: ~a\n"
+ mogrify status)))
+ (let ([next-procs (remq res procs)])
+ (if (not (empty? queue))
+ (begin
+ (let ([proc (start-job (first queue))])
+ (loop (rest queue) (cons proc next-procs))))
+ (unless (and (empty? next-procs) finish)
+ (loop queue next-procs))))]
+ [(eq? res mailbox)
+ (let ([msg (thread-receive)])
+ (cond
+ [(eq? msg 'finish)
+ (set! finish #t)
+ (unless (empty? procs)
+ (displayln "Waiting for ImageMagick processes to finish.")
+ (loop queue procs))]
+ [(job? msg)
+ (let ([j msg])
+ (if (>= (length procs) *max-jobs*)
+ (loop (append queue (list j)) procs) ; FIFO queue semantics
+ (let ([proc (start-job j)])
+ (loop queue (cons proc procs)))))]))]
+ [else
+ (error "Unknown sync result: " res)
+ (loop queue procs)])))))))
+
+(define/contract (resize-image input new-width out-path)
+ (path? number? path? . -> . void?)
+ (prn1 "Shrinking ~a to ~a pixels asynchronously." (abs->rel/www input) new-width)
+ ;; One problem with the async approach is that if Frog is killed before
+ ;; subprocesses are finished they will not be triggered again if Frog is
+ ;; invoked again and the source post has not been touched. Ideally we would
+ ;; trap SIGINT and write out unfinished work to disk, or at least
+ ;; detect that work was finished prematurely and clean and restart everything.
+ (thread-send master-worker (job input out-path new-width)))
+
+(define (wait-resize-images)
+ (thread-send master-worker 'finish)
+ (thread-wait master-worker))
+
+(module+ test
+ (when magick-available?
+ (parameterize ([top example]
+ [current-verbosity 0])
+ (define tmp (find-system-path 'temp-dir))
+ (define output (build-path tmp "600px-image.gif"))
+ (test-eq? "resize"
+ (begin
+ (resize-image (build-path (www/img-path) "600px-image.gif") 10 tmp)
+ (wait-resize-images)
+ (image-width output))
+ 10)
+ (delete-file* output))))
+
+(define/contract (get-images image-path)
+ (path? . -> . (values pair? (listof pair?)))
+ (define resized-dir (build-path* (www/img-path) (current-image-output-dir)))
+ (unless (directory-exists? resized-dir)
+ (make-directory resized-dir))
+ (let* ([orig-size (image-width image-path)]
+ [sizes (filter ((curry >) orig-size) (current-image-sizes))])
+ (values (cons image-path orig-size)
+ (append (for/list ([width sizes])
+ (define output-dir (build-path* resized-dir
+ (number->string width)))
+ (define output (build-path output-dir
+ (file-name-from-path image-path)))
+ (unless (directory-exists? output-dir)
+ (make-directory output-dir))
+ (unless (and (file-exists? output)
+ (< (file-or-directory-modify-seconds image-path)
+ (file-or-directory-modify-seconds output)))
+ ;; TODO Spawn asynchronously to enable utilizing more cores
+ (resize-image image-path width output-dir))
+ (cons output width))
+ (if (< (length sizes) (length (current-image-sizes)))
+ (list (cons image-path orig-size))
+ '())))))
+
+(define default-image-idx
+ (for/or ([v (current-image-sizes)]
+ [ix (in-naturals)])
+ (and (= v (current-image-default-size)) ix)))
+
+(define/contract (make-responsive path sizes)
+ (path-string? (or/c string? #f) . -> . (listof pair?))
+ (define image-path (build-path (www-path) (path->relative-path path)))
+ (define-values (orig srcset) (get-images image-path))
+ (define src (abs->rel/www (car (if (> (length srcset) default-image-idx)
+ (list-ref srcset default-image-idx)
+ orig))))
+ (define srcset-string
+ (string-join
+ (for/list ([srcdef srcset])
+ (format "~a ~aw" (~> (car srcdef)
+ abs->rel/www string->path
+ uri-encode-path path->string)
+ (cdr srcdef)))
+ ", "))
+ `([src ,src]
+ [srcset ,srcset-string]
+ ,(let ((orig-width (cdr orig)))
+ `[sizes ,(or sizes
+ (current-image-sizes-attr)
+ (format "(max-width: ~apx) 100vw, ~apx"
+ orig-width orig-width))])))
+
+(define/contract (clean-resized-images)
+ (-> any)
+ (let ([out-dir (build-path* (www/img-path) (current-image-output-dir))])
+ (when (directory-exists? out-dir)
+ (fold-files (λ (path type v)
+ (when (eq? type 'file)
+ (delete-file path)
+ (prn2 "Deleted ~a" (abs->rel/www path))))
+ '() out-dir #f)
+ (for-each delete-directory (directory-list out-dir #:build? #t))
+ (delete-directory out-dir))))
diff --git a/frog/util.rkt b/frog/util.rkt
index 79b32461..3874b02e 100644
--- a/frog/util.rkt
+++ b/frog/util.rkt
@@ -4,8 +4,9 @@
racket/function
racket/pretty
rackjure/threading
+ (only-in net/uri-codec uri-path-segment-encode)
(only-in markdown display-xexpr)
- "verbosity.rkt")
+ "verbosity.rkt")
(provide (all-defined-out))
@@ -68,3 +69,13 @@
(check-equal? (our-encode "Here's a question--how many hyphens???")
"Here-s-a-question-how-many-hyphens"))
+;; URI encode path to handle spaces and non-ascii characters
+(define (uri-encode-path path)
+ ;; (absolute-path? . -> . path?)
+ (let ([ps (for/list ([ps (explode-path path)])
+ (uri-path-segment-encode (path->string ps)))])
+ (apply build-path "/" (cdr ps))))
+
+(module+ test
+ (check-equal? (uri-encode-path (string->path "/dir/other dir/file name.ext"))
+ (string->path "/dir/other%20dir/file%20name.ext")))