You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
189 lines
8.5 KiB
189 lines
8.5 KiB
(ns NoteHub.views.pages |
|
(:require [NoteHub.crossover.lib :as lib] |
|
[clojure.contrib.string :as ccs] |
|
[hiccup.util :as util]) |
|
(:use |
|
[NoteHub.storage] |
|
[NoteHub.settings] |
|
[NoteHub.views.common] |
|
[clojure.string :rename {replace sreplace} :only [split replace lower-case]] |
|
[clojure.core.incubator :only [-?>]] |
|
[hiccup.form] |
|
[hiccup.core] |
|
[hiccup.element] |
|
[noir.response :only [redirect status content-type]] |
|
[noir.core :only [defpage defpartial]] |
|
[cheshire.core] |
|
[noir.statuses]) |
|
(:import |
|
[java.util Calendar] |
|
[org.pegdown PegDownProcessor])) |
|
|
|
; Create a new Object of the MD-processor |
|
(def md-processor |
|
(PegDownProcessor.)) |
|
|
|
; Markdown -> HTML mapper |
|
(defn md-to-html [md-text] |
|
(.markdownToHtml md-processor md-text)) |
|
|
|
; Sets a custom message for each needed HTTP status. |
|
; The message to be assigned is extracted with a dynamically generated key |
|
(doseq [code [400 403 404 500]] |
|
(set-page! code |
|
(let [message (get-message (keyword (str "status-" code)))] |
|
(layout message |
|
[:article [:h1 message]])))) |
|
|
|
; shortcut for rendering an HTTP status |
|
(defn- response [code] |
|
(status code (get-page code))) |
|
|
|
; Converts given markdown to html and wraps with the main layout |
|
(defn- wrap [short-url params md-text] |
|
(when md-text |
|
(layout params (params :title) |
|
[:article.bottom-space (md-to-html md-text)] |
|
(let [links (map #(link-to |
|
(if (= :short-url %) |
|
(url short-url) |
|
(str (params :title) "/" (name %))) |
|
(get-message %)) |
|
[:stats :edit :export :short-url]) |
|
space (apply str (repeat 4 " ")) |
|
separator (str space "·" space) |
|
links (interpose separator links)] |
|
[:div#panel (map identity links)])))) |
|
|
|
(defn get-date |
|
"Returns today's date" |
|
[] |
|
(map #(+ (second %) (.get (Calendar/getInstance) (first %))) |
|
{Calendar/YEAR 0, Calendar/MONTH 1, Calendar/DAY_OF_MONTH 0})) |
|
|
|
; Routes |
|
; ====== |
|
|
|
; This function answers to an AJAX request: it gets a session key and a markdown text. |
|
; It returns the html code of the provided markdown and a new session key. |
|
(defpage [:post "/preview"] {:keys [session-key draft]} |
|
(generate-string |
|
{:preview (md-to-html draft)})) |
|
|
|
; Landing Page |
|
(defpage "/" {} |
|
(layout (get-message :title) |
|
[:div#hero |
|
[:h1 (get-message :name)] |
|
[:h2 (get-message :title)] |
|
[:br] |
|
[:a.landing-button {:href "/new" :style "color: white"} (get-message :new-page)]] |
|
[:div#dashed-line] |
|
; dynamically generates three column, retrieving corresponding messages |
|
[:article.helvetica-neue.bottom-space {:style "font-size: 1em"} |
|
(md-to-html (slurp "README.md"))] |
|
[:div.centered.helvetica-neue (md-to-html (get-message :created-by))])) |
|
|
|
; input form for the markdown text with a preview area |
|
(defpartial input-form [form-url command fields content passwd-msg] |
|
(let [css-class (when (= :publish command) :hidden)] |
|
(layout {:js true} (get-message :new-note) |
|
[:article#preview " "] |
|
[:div#dashed-line {:class css-class}] |
|
[:div.central-element.helvetica-neue {:style "margin-bottom: 3em"} |
|
(form-to {:autocomplete :off} [:post form-url] |
|
(hidden-field :action command) |
|
(hidden-field :password) |
|
fields |
|
(text-area {:class :max-width} :draft content) |
|
[:fieldset#input-elems {:class css-class} |
|
(get-message passwd-msg) |
|
(text-field {:class "ui-elem"} :plain-password) |
|
(submit-button {:class "button ui-elem" |
|
:id :publish-button} (get-message command))])]))) |
|
|
|
; Update Note Page |
|
(defpage "/:year/:month/:day/:title/edit" {:keys [year month day title]} |
|
(input-form "/update-note" :update |
|
(html (hidden-field :key (build-key [year month day] title))) |
|
(get-note [year month day] title) :enter-passwd)) |
|
|
|
; New Note Page |
|
(defpage "/new" {} |
|
(input-form "/post-note" :publish |
|
(html (hidden-field :session-key (create-session)) |
|
(hidden-field {:id :session-value} :session-value)) |
|
(get-message :loading) :set-passwd)) |
|
|
|
; Displays the note |
|
(defpage "/:year/:month/:day/:title" {:keys [year month day title theme header-font text-font] :as params} |
|
(wrap |
|
(create-short-url params) |
|
(select-keys params [:title :theme :header-font :text-font]) |
|
(get-note [year month day] title))) |
|
|
|
; Provides Markdown of the specified note |
|
(defpage "/:year/:month/:day/:title/export" {:keys [year month day title]} |
|
(when-let [md-text (get-note [year month day] title)] |
|
(content-type "text/plain; charset=utf-8" md-text))) |
|
|
|
; Provides the number of views of the specified note |
|
(defpage "/:year/:month/:day/:title/stats" {:keys [year month day title]} |
|
(when-let [views (get-note-views [year month day] title)] |
|
(layout (get-message :statistics) |
|
[:table#stats.helvetica-neue.central-element |
|
[:tr |
|
[:td (get-message :published)] |
|
[:td (interpose "-" [year month day])]] |
|
[:tr |
|
[:td (get-message :article-views)] |
|
[:td views]]]))) |
|
|
|
; Updates a note |
|
(defpage [:post "/update-note"] {:keys [key draft password]} |
|
(if (update-note key draft password) |
|
(redirect (apply url (split key #" "))) |
|
(response 403))) |
|
|
|
; New Note Posting — the most "complex" function in the entire app ;) |
|
(defpage [:post "/post-note"] {:keys [draft password session-key session-value]} |
|
; first we collect all info needed to evaluate the validity of the note creation request |
|
(let [valid-session (invalidate-session session-key) ; was the note posted from a newly generated form? |
|
valid-draft (not (ccs/blank? draft)) ; has the note a meaningful content? |
|
; is the hash code correct? |
|
valid-hash (try |
|
(= (Short/parseShort session-value) |
|
(lib/hash #(.codePointAt % 0) (str draft session-key))) |
|
(catch Exception e nil))] |
|
; check whether the new note can be added |
|
(if (and valid-session valid-draft valid-hash) |
|
; if yes, we compute the current date, extract a title string from the text, |
|
; which will be a part of the url and look whether this title is free today; |
|
; if not, append "-n", where "n" is the next free number |
|
(let [[year month day] (get-date) |
|
untrimmed-line (filter #(or (= \- %) (Character/isLetterOrDigit %)) |
|
(-> draft ccs/split-lines first (sreplace " " "-") lower-case)) |
|
trim (fn [s] (apply str (drop-while #(= \- %) s))) |
|
title-uncut (-> untrimmed-line trim reverse trim reverse) |
|
max-length (get-setting :max-title-length #(Integer/parseInt %) 80) |
|
; TODO: replace to ccs/take when it gets fixed |
|
proposed-title (apply str (take max-length title-uncut)) |
|
date [year month day] |
|
title (first (drop-while #(note-exists? date %) |
|
(cons proposed-title |
|
(map #(str proposed-title "-" (+ 2 %)) (range)))))] |
|
(do |
|
(set-note date title draft password) |
|
(redirect (url year month day title)))) |
|
(response 400)))) |
|
|
|
; Resolving of a short url |
|
(defpage "/:short-url" {:keys [short-url]} |
|
(when-let [params (resolve-url short-url)] |
|
(let [{:keys [year month day title]} params |
|
rest-params (dissoc params :year :month :day :title) |
|
core-url (url year month day title) |
|
long-url (if (empty? rest-params) core-url (util/url core-url rest-params))] |
|
(redirect long-url)))) |
|
|
|
|
|
|