reagent/demo/sitetools/core.cljs

115 lines
3.8 KiB
Plaintext
Raw Normal View History

(ns sitetools.core
(:require [clojure.string :as string]
[goog.events :as evt]
[reagent.core :as r]
[reagent.debug :refer-macros [dbg log dev?]])
2015-09-08 00:41:04 +02:00
(:import goog.History
2015-09-16 17:55:45 +02:00
[goog.history Html5History EventType]))
2015-09-16 17:55:45 +02:00
(enable-console-print!)
;;; Configuration
2015-09-09 12:11:30 +02:00
(declare main-content)
2015-09-08 00:29:06 +02:00
2015-09-09 12:11:30 +02:00
(defonce config (r/atom {:body [#'main-content]
2015-09-15 12:27:07 +02:00
:pages {"/index.html" {:content [:div]
:title ""}}
:site-dir "target/prerender/public/"
:css-infiles ["site/public/css/main.css"]
:css-file "css/built.css"
:js-file "js/main.js"
:main-div "main-content"
2015-09-08 13:08:09 +02:00
:default-title ""}))
2015-09-07 22:22:26 +02:00
(defonce history nil)
2015-09-15 13:36:02 +02:00
(defn demo-handler [state [id x :as event]]
2015-09-07 22:22:26 +02:00
(case id
2015-09-15 13:36:02 +02:00
:set-content (let [page x
title (:title page)
title (if title
(str (:title-prefix state) title)
2015-09-16 17:55:45 +02:00
(str (:default-title state)))]
2015-09-08 18:07:29 +02:00
(when r/is-client
2015-09-09 11:46:41 +02:00
(set! js/document.title title))
2015-09-16 17:55:45 +02:00
(assoc state :current-page page :title title))
2015-09-15 13:36:02 +02:00
:set-page (let [path x
_ (assert (string? path))
ps (:pages state)
p (get ps path (get ps "/index.html"))]
(recur state [:set-content p]))
:goto-page (let [path x
_ (assert (string? path))]
2015-09-15 11:57:12 +02:00
(when-some [h history]
2016-05-02 12:44:02 +02:00
(r/after-render (fn []
(.setToken h x)
(js/scrollTo 0 0)))
2015-09-15 11:57:12 +02:00
state)
(recur state [:set-page x]))))
2015-09-07 22:22:26 +02:00
(defn emit [event]
2015-09-08 13:08:09 +02:00
;; (dbg event)
(r/rswap! config demo-handler event))
2015-09-07 22:22:26 +02:00
2015-09-08 20:18:27 +02:00
(defn register-page [url comp title]
{:pre [(re-matches #"/.*[.]html" url)
(vector? comp)]}
2015-09-15 12:27:07 +02:00
(swap! config update-in [:pages]
assoc url {:content comp :title title}))
2015-09-08 20:18:27 +02:00
2015-09-08 13:08:09 +02:00
;;; History
2015-09-09 10:51:37 +02:00
(defn init-history [page]
2015-09-07 22:22:26 +02:00
(when-not history
2015-09-09 10:51:37 +02:00
(let [html5 (and page
2015-09-09 16:00:21 +02:00
(Html5History.isSupported)
2015-09-08 13:08:09 +02:00
(#{"http:" "https:"} js/location.protocol))]
2015-09-08 09:35:55 +02:00
(doto (set! history
(if html5
(doto (Html5History.)
(.setUseFragment false)
2015-09-08 13:08:09 +02:00
(.setPathPrefix (-> js/location.pathname
(string/replace
(re-pattern (str page "$")) "")
(string/replace #"/*$" ""))))
2015-09-08 09:35:55 +02:00
(History.)))
2015-09-16 17:55:45 +02:00
(evt/listen EventType.NAVIGATE #(when (.-isNavigation %)
2016-05-02 12:44:02 +02:00
(emit [:set-page (.-token %)])
(r/flush)))
2015-09-08 09:35:55 +02:00
(.setEnabled true))
2015-09-16 17:55:45 +02:00
(let [token (.getToken history)
p (if (and page (not html5) (empty? token))
2015-09-15 11:57:12 +02:00
page
2015-09-16 17:55:45 +02:00
token)]
(emit [:set-page p])))))
2015-09-07 22:22:26 +02:00
2015-09-08 18:00:44 +02:00
(defn to-relative [f]
2015-09-08 13:08:09 +02:00
(string/replace f #"^/" ""))
;;; Components
2015-09-07 22:22:26 +02:00
(defn link [props child]
[:a (assoc props
2015-09-08 18:00:44 +02:00
:href (-> props :href to-relative)
2015-09-07 22:22:26 +02:00
:on-click #(do (.preventDefault %)
(emit [:goto-page (:href props)])))
2015-09-07 22:22:26 +02:00
child])
2015-09-09 12:11:30 +02:00
(defn main-content []
2015-09-16 17:55:45 +02:00
(get-in @config [:current-page :content]))
;;; Main entry points
(defn start! [site-config]
(swap! config merge site-config)
(when r/is-client
2015-09-09 10:51:37 +02:00
(let [page-conf (when (exists? js/pageConfig)
(js->clj js/pageConfig :keywordize-keys true))
2015-09-09 11:46:41 +02:00
conf (swap! config merge page-conf)
2015-09-15 12:27:07 +02:00
{:keys [page-path body main-div]} conf]
(init-history page-path)
(r/render body (js/document.getElementById main-div)))))