2016-03-15 15:35:35 +03:00
|
|
|
|
{-# LANGUAGE
|
|
|
|
|
QuasiQuotes,
|
|
|
|
|
OverloadedStrings,
|
|
|
|
|
FlexibleContexts,
|
2016-03-16 02:17:08 +03:00
|
|
|
|
ViewPatterns,
|
2016-04-07 22:14:08 +03:00
|
|
|
|
RecordWildCards,
|
2016-04-15 14:14:01 +03:00
|
|
|
|
TupleSections,
|
2016-03-15 15:35:35 +03:00
|
|
|
|
NoImplicitPrelude
|
|
|
|
|
#-}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
module View
|
|
|
|
|
(
|
|
|
|
|
-- * Pages
|
|
|
|
|
renderRoot,
|
2016-04-07 22:14:08 +03:00
|
|
|
|
renderAdmin,
|
2016-04-15 14:14:01 +03:00
|
|
|
|
renderEdits,
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderHaskellRoot,
|
2016-03-15 15:35:35 +03:00
|
|
|
|
renderDonate,
|
2016-03-19 02:40:00 +03:00
|
|
|
|
renderCategoryPage,
|
2016-03-20 13:50:23 +03:00
|
|
|
|
renderUnwrittenRules,
|
2016-05-02 21:27:49 +03:00
|
|
|
|
renderMarkdownHelp,
|
2016-05-01 23:17:55 +03:00
|
|
|
|
renderSearchResults,
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
-- * Tracking
|
|
|
|
|
renderTracking,
|
|
|
|
|
|
|
|
|
|
-- * Methods
|
|
|
|
|
-- ** Categories
|
|
|
|
|
renderCategoryList,
|
|
|
|
|
renderCategory,
|
2016-05-01 23:17:55 +03:00
|
|
|
|
renderCategoryHeader,
|
2016-03-15 15:35:35 +03:00
|
|
|
|
renderCategoryNotes,
|
|
|
|
|
-- ** Items
|
|
|
|
|
renderItem,
|
|
|
|
|
renderItemInfo,
|
|
|
|
|
renderItemDescription,
|
2016-03-17 02:52:40 +03:00
|
|
|
|
renderItemEcosystem,
|
2016-03-15 15:35:35 +03:00
|
|
|
|
renderItemTraits,
|
|
|
|
|
renderItemNotes,
|
|
|
|
|
-- ** Traits
|
|
|
|
|
renderTrait,
|
|
|
|
|
|
2016-03-19 21:36:21 +03:00
|
|
|
|
-- * Rendering for feeds
|
|
|
|
|
renderItemForFeed,
|
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- * Miscellaneous
|
|
|
|
|
getItemHue,
|
|
|
|
|
newGroupValue,
|
|
|
|
|
)
|
|
|
|
|
where
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
-- General
|
|
|
|
|
import BasePrelude hiding (Category)
|
2016-04-16 02:02:43 +03:00
|
|
|
|
-- Default
|
|
|
|
|
import Data.Default
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- Lenses
|
|
|
|
|
import Lens.Micro.Platform hiding ((&))
|
|
|
|
|
-- Monads and monad transformers
|
|
|
|
|
import Control.Monad.IO.Class
|
2016-03-22 01:17:53 +03:00
|
|
|
|
import Control.Monad.Reader
|
2016-04-21 16:47:14 +03:00
|
|
|
|
import Control.Monad.Random
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- Containers
|
|
|
|
|
import qualified Data.Map as M
|
2016-04-16 02:02:43 +03:00
|
|
|
|
import Data.Tree
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- Text
|
|
|
|
|
import qualified Data.Text as T
|
|
|
|
|
import qualified Data.Text.IO as T
|
|
|
|
|
import Data.Text (Text)
|
|
|
|
|
import NeatInterpolation
|
|
|
|
|
-- Web
|
|
|
|
|
import Lucid hiding (for_)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
-- Time
|
|
|
|
|
import Data.Time.Format.Human
|
2016-04-16 02:02:43 +03:00
|
|
|
|
-- Markdown
|
|
|
|
|
import Cheapskate.Lucid
|
|
|
|
|
import Cheapskate.Types
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
-- Local
|
2016-03-22 01:17:53 +03:00
|
|
|
|
import Config
|
2016-03-15 15:35:35 +03:00
|
|
|
|
import Types
|
|
|
|
|
import Utils
|
|
|
|
|
import JS (JS(..), JQuerySelector)
|
|
|
|
|
import qualified JS
|
|
|
|
|
import Markdown
|
2016-05-01 16:28:10 +03:00
|
|
|
|
import Cache
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{- Note [autosize]
|
|
|
|
|
~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
|
|
All textareas on the site are autosized – i.e. they grow when the user is typing. This is done by the autosize.js plugin, which is called on page load:
|
|
|
|
|
|
|
|
|
|
autosize($('textarea'));
|
|
|
|
|
|
|
|
|
|
A slight problem is that it doesn't compute the height of hidden elements correctly – thus, when something is shown and it happens to be a textarea or contain a textarea, we have to call autosize again. This is done in 'JS.switchSection'. So far there are no textboxes that are shown *without* switchSection being involved, and so there's no need to watch for elements being added to the DOM.
|
|
|
|
|
|
|
|
|
|
It would be nicer if we could watch for elements becoming visible without having to modify switchSection, but there doesn't seem to be an easy way to do this – MutationObserver doesn't let us find out when something becomes visible (i.e. when its clientHeight stops being 0).
|
|
|
|
|
|
|
|
|
|
In switchSection we use
|
|
|
|
|
|
|
|
|
|
autosize($('textarea'));
|
|
|
|
|
autosize.update($('textarea'));
|
|
|
|
|
|
|
|
|
|
instead of simple
|
|
|
|
|
|
|
|
|
|
autosize.update($('textarea'));
|
|
|
|
|
|
|
|
|
|
– this is done because the textarea could have appeared after the original `autosize($('textarea'));` was called on page load (which could happen if an item was added, for instance).
|
|
|
|
|
|
|
|
|
|
-}
|
|
|
|
|
|
2016-03-27 02:34:07 +03:00
|
|
|
|
{- Note [show-hide]
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
2016-05-02 21:44:50 +03:00
|
|
|
|
A lot of things (notes, etc) can be expanded/collapsed by pressing a button. Similarly, pressing “edit” replaces rendered text with a textbox, or adds buttons to pros/cons. All this is done with sections and show/hide.
|
2016-03-27 02:34:07 +03:00
|
|
|
|
|
|
|
|
|
A section is something that can be shown or hidden. You define a section by using 'section' (which creates a <div>) or 'sectionSpan' (which creates a <span>).
|
|
|
|
|
|
|
|
|
|
section "normal" [shown, noScriptShown] $ do
|
|
|
|
|
renderText
|
|
|
|
|
...
|
|
|
|
|
|
|
|
|
|
section "editing" [] $ do
|
|
|
|
|
renderEditbox
|
|
|
|
|
...
|
|
|
|
|
|
2016-04-11 23:31:30 +03:00
|
|
|
|
You can even give 2 names to a section – e.g. "normal editing" if you want the section be visible both in “normal” mode and in “editing” mode.
|
|
|
|
|
|
2016-03-27 02:34:07 +03:00
|
|
|
|
The list parameter is used to add attributes to the section. 'shown' is an attribute that means that the section is normally visible; 'noScriptShown' means that the section will be visible when Javascipt is disabled. Sections without either attribute will be hidden. (Usually 'shown' and 'noScriptShown' go together, but not always.)
|
|
|
|
|
|
|
|
|
|
When several sections are in the same container (e.g. a <div>), you can toggle between them with 'JS.switchSection', which shows the section (or several sections) with given name, and hides all sections with other names. The elements that aren't sections are not affected.
|
|
|
|
|
|
|
|
|
|
Also, there's another function available – 'JS.switchSectionEverywhere' – that switches sections everywhere inside the container, not only among container's direct children. It's useful when you have something like a list of pros/cons and you want to switch them all into the “editable” state.
|
|
|
|
|
|
2016-04-11 23:31:30 +03:00
|
|
|
|
////////////////////////////////////
|
|
|
|
|
|
2016-03-27 02:34:07 +03:00
|
|
|
|
And now, here's how it's all implemented.
|
|
|
|
|
|
|
|
|
|
In 'wrapPage' there's a piece of CSS wrapped in <noscript> that hides everything except for 'noScriptShown' things:
|
|
|
|
|
|
|
|
|
|
.section:not(.noscript-shown) {display:none;}
|
|
|
|
|
|
|
|
|
|
There's also a piece of Javascript that, when executed, will change it to the following CSS:
|
|
|
|
|
|
|
|
|
|
.section:not(.shown) {display:none;}
|
|
|
|
|
|
|
|
|
|
So, if Javascript is disabled we hide all sections except for those that have the 'noScriptShown' attribute, and if it's enabled we hide all sections except for those that have the 'shown' attribute.
|
|
|
|
|
|
|
|
|
|
After that switching sections is simply done by adding/removing the “shown” class. (Note that we don't have to choose between “noscript-shown” and “shown” because switching sections is *only* possible if Javascript is enabled, and in this case the relevant tag will always be “shown” and not “noscript-shown”.)
|
|
|
|
|
|
|
|
|
|
-}
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderRoot :: (MonadIO m, MonadRandom m, MonadReader Config m) => HtmlT m ()
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderRoot = do
|
|
|
|
|
wrapPage "Aelve Guide" $ do
|
|
|
|
|
h1_ "Aelve Guide"
|
2016-05-03 03:27:30 +03:00
|
|
|
|
h2_ (mkLink "Haskell" "/haskell")
|
2016-03-23 18:28:03 +03:00
|
|
|
|
|
2016-04-07 22:14:08 +03:00
|
|
|
|
-- TODO: show a “category not found” page
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderAdmin
|
|
|
|
|
:: (MonadIO m, MonadRandom m)
|
|
|
|
|
=> GlobalState -> [(Edit, EditDetails)] -> HtmlT m ()
|
2016-04-07 22:14:08 +03:00
|
|
|
|
renderAdmin globalState edits = do
|
2016-04-08 18:05:52 +03:00
|
|
|
|
head_ $ do
|
|
|
|
|
includeJS "/js.js"
|
|
|
|
|
includeJS "/jquery-2.2.0.min.js"
|
2016-04-09 03:36:38 +03:00
|
|
|
|
includeCSS "/markup.css"
|
2016-04-14 01:56:13 +03:00
|
|
|
|
includeCSS "/admin.css"
|
2016-04-16 23:06:59 +03:00
|
|
|
|
includeCSS "/loader.css"
|
2016-04-13 18:38:35 +03:00
|
|
|
|
title_ "admin – Aelve Guide"
|
2016-04-08 18:05:52 +03:00
|
|
|
|
meta_ [name_ "viewport",
|
|
|
|
|
content_ "width=device-width, initial-scale=1.0, user-scalable=yes"]
|
|
|
|
|
|
|
|
|
|
body_ $ do
|
2016-04-16 23:06:59 +03:00
|
|
|
|
script_ $ fromJS $ JS.createAjaxIndicator ()
|
2016-04-16 00:06:34 +03:00
|
|
|
|
h1_ "Miscellaneous"
|
|
|
|
|
buttonUid <- randomLongUid
|
|
|
|
|
button "Create checkpoint" [uid_ buttonUid] $
|
|
|
|
|
JS.createCheckpoint [JS.selectUid buttonUid]
|
2016-04-13 18:38:35 +03:00
|
|
|
|
h1_ "Pending edits"
|
2016-04-15 14:14:01 +03:00
|
|
|
|
renderEdits globalState (map (,Nothing) edits)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
2016-04-09 00:50:13 +03:00
|
|
|
|
-- TODO: when showing Edit'DeleteCategory, show the amount of items in that
|
|
|
|
|
-- category and titles of items themselves
|
|
|
|
|
|
2016-04-15 14:14:01 +03:00
|
|
|
|
-- | Group edits by IP and render them
|
|
|
|
|
renderEdits
|
2016-04-21 16:47:14 +03:00
|
|
|
|
:: (MonadIO m, MonadRandom m)
|
2016-04-15 14:14:01 +03:00
|
|
|
|
=> GlobalState
|
|
|
|
|
-> [((Edit, EditDetails), Maybe String)]
|
|
|
|
|
-> HtmlT m ()
|
|
|
|
|
renderEdits globalState edits = do
|
|
|
|
|
let editBlocks = groupBy (equating (editIP . snd . fst)) edits
|
|
|
|
|
for_ editBlocks $ \editBlock -> div_ $ do
|
|
|
|
|
blockNode <- thisNode
|
|
|
|
|
h2_ $ do
|
|
|
|
|
case editIP (editBlock ^?! _head._1._2) of
|
|
|
|
|
Nothing -> "<unknown IP>"
|
|
|
|
|
Just ip -> toHtml (show ip)
|
|
|
|
|
emptySpan "1em"
|
|
|
|
|
textButton "accept all" $
|
|
|
|
|
JS.acceptBlock (editId (editBlock ^?! _head._1._2),
|
|
|
|
|
editId (editBlock ^?! _last._1._2),
|
|
|
|
|
blockNode)
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
textButton "undo all" $
|
|
|
|
|
JS.undoBlock (editId (editBlock ^?! _head._1._2),
|
|
|
|
|
editId (editBlock ^?! _last._1._2),
|
|
|
|
|
blockNode)
|
|
|
|
|
ul_ $ do
|
|
|
|
|
for_ editBlock $ \((edit, EditDetails{..}), mbErr) -> li_ $ do
|
|
|
|
|
editNode <- thisNode
|
|
|
|
|
p_ $ do
|
|
|
|
|
toHtml =<< liftIO (humanReadableTime editDate)
|
|
|
|
|
emptySpan "1em"
|
|
|
|
|
textButton "accept" $
|
|
|
|
|
JS.acceptEdit (editId, editNode)
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
textButton "try to undo" $
|
|
|
|
|
JS.undoEdit (editId, editNode)
|
|
|
|
|
case mbErr of
|
|
|
|
|
Nothing -> return ()
|
|
|
|
|
Just err -> p_ $ span_ [style_ "background-color:#E57373"] $
|
|
|
|
|
"Can't apply the edit: " >> toHtml err
|
|
|
|
|
renderEdit globalState edit
|
|
|
|
|
|
2016-04-07 22:14:08 +03:00
|
|
|
|
renderEdit :: Monad m => GlobalState -> Edit -> HtmlT m ()
|
|
|
|
|
renderEdit globalState edit = do
|
|
|
|
|
let quote :: Monad m => HtmlT m () -> HtmlT m ()
|
|
|
|
|
quote a = "“" *> a <* "”"
|
|
|
|
|
-- We're searching for everything (items/categories) both in normal lists
|
|
|
|
|
-- and in lists of deleted things. Just in case.
|
|
|
|
|
let allCategories = globalState^.categories ++
|
|
|
|
|
globalState^.categoriesDeleted
|
2016-04-09 00:50:13 +03:00
|
|
|
|
let findCategory catId = fromMaybe err (find (hasUid catId) allCategories)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
where
|
|
|
|
|
err = error ("renderEdit: couldn't find category with uid = " ++
|
|
|
|
|
T.unpack (uidToText catId))
|
|
|
|
|
let findItem itemId = (category, item)
|
|
|
|
|
where
|
|
|
|
|
getItems = view (items <> itemsDeleted)
|
2016-04-09 00:50:13 +03:00
|
|
|
|
ourCategory = any (hasUid itemId) . getItems
|
2016-04-07 22:14:08 +03:00
|
|
|
|
err = error ("renderEdit: couldn't find item with uid = " ++
|
|
|
|
|
T.unpack (uidToText itemId))
|
|
|
|
|
category = fromMaybe err (find ourCategory allCategories)
|
2016-04-09 00:50:13 +03:00
|
|
|
|
item = fromJust (find (hasUid itemId) (getItems category))
|
2016-04-07 22:14:08 +03:00
|
|
|
|
let findTrait itemId traitId = (category, item, trait)
|
|
|
|
|
where
|
|
|
|
|
(category, item) = findItem itemId
|
|
|
|
|
getTraits = view (cons <> consDeleted <> pros <> prosDeleted)
|
|
|
|
|
err = error ("renderEdit: couldn't find trait with uid = " ++
|
|
|
|
|
T.unpack (uidToText traitId))
|
2016-04-09 00:50:13 +03:00
|
|
|
|
trait = fromMaybe err (find (hasUid traitId) (getTraits item))
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
let printCategory catId = do
|
|
|
|
|
let category = findCategory catId
|
|
|
|
|
quote $ toHtml (category ^. title)
|
|
|
|
|
let printItem itemId = do
|
|
|
|
|
let (_, item) = findItem itemId
|
|
|
|
|
quote $ toHtml (item ^. name)
|
|
|
|
|
|
|
|
|
|
case edit of
|
|
|
|
|
-- Add
|
|
|
|
|
Edit'AddCategory _catId title' -> p_ $ do
|
|
|
|
|
"added category " >> quote (toHtml title')
|
|
|
|
|
Edit'AddItem catId _itemId name' -> p_ $ do
|
|
|
|
|
"added item " >> quote (toHtml name')
|
|
|
|
|
" to category " >> printCategory catId
|
|
|
|
|
Edit'AddPro itemId _traitId content' -> do
|
|
|
|
|
p_ $ "added pro to item " >> printItem itemId
|
2016-04-14 01:56:13 +03:00
|
|
|
|
blockquote_ $ p_ $ toHtml (renderMarkdownInline content')
|
2016-04-07 22:14:08 +03:00
|
|
|
|
Edit'AddCon itemId _traitId content' -> do
|
|
|
|
|
p_ $ "added con to item " >> printItem itemId
|
2016-04-14 01:56:13 +03:00
|
|
|
|
blockquote_ $ p_ $ toHtml (renderMarkdownInline content')
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
-- Change category properties
|
|
|
|
|
Edit'SetCategoryTitle _catId oldTitle newTitle -> p_ $ do
|
|
|
|
|
"changed title of category " >> quote (toHtml oldTitle)
|
|
|
|
|
" to " >> quote (toHtml newTitle)
|
2016-05-01 23:17:55 +03:00
|
|
|
|
Edit'SetCategoryGroup catId oldGroup newGroup -> p_ $ do
|
|
|
|
|
"changed group of category " >> printCategory catId
|
|
|
|
|
" from " >> quote (toHtml oldGroup)
|
|
|
|
|
" to " >> quote (toHtml newGroup)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
Edit'SetCategoryNotes catId oldNotes newNotes -> do
|
|
|
|
|
p_ $ "changed notes of category " >> printCategory catId
|
|
|
|
|
table_ $ tr_ $ do
|
2016-04-14 01:56:13 +03:00
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock oldNotes)
|
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock newNotes)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
-- Change item properties
|
|
|
|
|
Edit'SetItemName _itemId oldName newName -> p_ $ do
|
|
|
|
|
"changed name of item " >> quote (toHtml oldName)
|
|
|
|
|
" to " >> quote (toHtml newName)
|
|
|
|
|
Edit'SetItemLink itemId oldLink newLink -> p_ $ do
|
|
|
|
|
"changed link of item " >> printItem itemId
|
|
|
|
|
" from " >> code_ (toHtml (show oldLink))
|
|
|
|
|
" to " >> code_ (toHtml (show newLink))
|
|
|
|
|
Edit'SetItemGroup itemId oldGroup newGroup -> p_ $ do
|
|
|
|
|
"changed group of item " >> printItem itemId
|
|
|
|
|
" from " >> code_ (toHtml (show oldGroup))
|
|
|
|
|
" to " >> code_ (toHtml (show newGroup))
|
|
|
|
|
Edit'SetItemKind itemId oldKind newKind -> p_ $ do
|
|
|
|
|
"changed kind of item " >> printItem itemId
|
|
|
|
|
" from " >> code_ (toHtml (show oldKind))
|
|
|
|
|
" to " >> code_ (toHtml (show newKind))
|
|
|
|
|
Edit'SetItemDescription itemId oldDescr newDescr -> do
|
|
|
|
|
p_ $ "changed description of item " >> printItem itemId
|
|
|
|
|
table_ $ tr_ $ do
|
2016-04-14 01:56:13 +03:00
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock oldDescr)
|
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock newDescr)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
Edit'SetItemNotes itemId oldNotes newNotes -> do
|
|
|
|
|
p_ $ "changed notes of item " >> printItem itemId
|
|
|
|
|
table_ $ tr_ $ do
|
2016-04-14 01:56:13 +03:00
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock oldNotes)
|
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock newNotes)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
Edit'SetItemEcosystem itemId oldEcosystem newEcosystem -> do
|
|
|
|
|
p_ $ "changed ecosystem of item " >> printItem itemId
|
|
|
|
|
table_ $ tr_ $ do
|
2016-04-14 01:56:13 +03:00
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock oldEcosystem)
|
|
|
|
|
td_ $ blockquote_ $ toHtml (renderMarkdownBlock newEcosystem)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
-- Change trait properties
|
|
|
|
|
Edit'SetTraitContent itemId _traitId oldContent newContent -> do
|
|
|
|
|
p_ $ "changed trait of item " >> printItem itemId
|
|
|
|
|
table_ $ tr_ $ do
|
2016-04-14 01:56:13 +03:00
|
|
|
|
td_ $ blockquote_ $ p_ (toHtml (renderMarkdownInline oldContent))
|
|
|
|
|
td_ $ blockquote_ $ p_ (toHtml (renderMarkdownInline newContent))
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
-- Delete
|
|
|
|
|
Edit'DeleteCategory catId _pos -> p_ $ do
|
|
|
|
|
"deleted category " >> printCategory catId
|
|
|
|
|
Edit'DeleteItem itemId _pos -> p_ $ do
|
|
|
|
|
let (category, item) = findItem itemId
|
|
|
|
|
"deleted item " >> quote (toHtml (item^.name))
|
|
|
|
|
" from category " >> quote (toHtml (category^.title))
|
|
|
|
|
Edit'DeleteTrait itemId traitId _pos -> do
|
|
|
|
|
let (_, item, trait) = findTrait itemId traitId
|
|
|
|
|
p_ $ "deleted trait from item " >> quote (toHtml (item^.name))
|
2016-04-14 01:56:13 +03:00
|
|
|
|
blockquote_ $ p_ $ toHtml (trait^.content)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
-- Other
|
|
|
|
|
Edit'MoveItem itemId direction -> p_ $ do
|
|
|
|
|
"moved item " >> printItem itemId
|
|
|
|
|
if direction then " up" else " down"
|
|
|
|
|
Edit'MoveTrait itemId traitId direction -> do
|
|
|
|
|
let (_, item, trait) = findTrait itemId traitId
|
|
|
|
|
p_ $ "moved trait of item " >> quote (toHtml (item^.name)) >>
|
|
|
|
|
if direction then " up" else " down"
|
2016-04-14 01:56:13 +03:00
|
|
|
|
blockquote_ $ p_ $ toHtml (trait^.content)
|
2016-04-07 22:14:08 +03:00
|
|
|
|
|
|
|
|
|
-- TODO: use “data Direction = Up | Down” for directions instead of Bool
|
|
|
|
|
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderHaskellRoot
|
2016-04-21 16:47:14 +03:00
|
|
|
|
:: (MonadIO m, MonadRandom m, MonadReader Config m)
|
2016-03-22 01:17:53 +03:00
|
|
|
|
=> GlobalState -> Maybe Text -> HtmlT m ()
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderHaskellRoot globalState mbSearchQuery =
|
2016-03-19 02:40:00 +03:00
|
|
|
|
wrapPage "Aelve Guide" $ do
|
2016-04-20 01:59:29 +03:00
|
|
|
|
onPageLoad $ JS.expandHash ()
|
2016-04-15 23:44:55 +03:00
|
|
|
|
case mbSearchQuery of
|
|
|
|
|
Nothing -> h1_ "The Haskeller's guide"
|
|
|
|
|
-- A search page isn't the main page, so we need a link to the main page
|
2016-05-03 03:27:30 +03:00
|
|
|
|
Just _ -> h1_ (mkLink "The Haskeller's guide" "/haskell")
|
2016-04-06 01:36:55 +03:00
|
|
|
|
renderNoScriptWarning
|
2016-04-21 17:01:54 +03:00
|
|
|
|
renderSearch mbSearchQuery
|
2016-03-15 15:35:35 +03:00
|
|
|
|
textInput [
|
|
|
|
|
placeholder_ "add a category",
|
|
|
|
|
autocomplete_ "off",
|
2016-04-21 18:10:58 +03:00
|
|
|
|
onEnter $ JS.addCategoryAndRedirect [inputValue] ]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
case mbSearchQuery of
|
|
|
|
|
Nothing -> renderCategoryList (globalState^.categories)
|
|
|
|
|
Just query' -> do
|
|
|
|
|
let queryWords = T.words query'
|
|
|
|
|
let rank :: Category -> Int
|
|
|
|
|
rank cat = sum [
|
|
|
|
|
length (queryWords `intersect` (cat^..items.each.name)),
|
|
|
|
|
length (queryWords `intersect` T.words (cat^.title)) ]
|
|
|
|
|
let rankedCategories
|
|
|
|
|
| null queryWords = globalState^.categories
|
|
|
|
|
| otherwise = filter ((/= 0) . rank) .
|
|
|
|
|
reverse . sortOn rank
|
|
|
|
|
$ globalState^.categories
|
2016-05-01 23:17:55 +03:00
|
|
|
|
renderSearchResults rankedCategories
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- TODO: maybe add a button like “give me random category that is
|
|
|
|
|
-- unfinished”
|
|
|
|
|
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderCategoryPage
|
2016-04-21 16:47:14 +03:00
|
|
|
|
:: (MonadIO m, MonadRandom m, MonadReader Config m)
|
|
|
|
|
=> Category -> HtmlT m ()
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderCategoryPage category = do
|
2016-03-23 18:28:03 +03:00
|
|
|
|
wrapPage (category^.title <> " – Aelve Guide") $ do
|
2016-04-20 01:59:29 +03:00
|
|
|
|
onPageLoad $ JS.expandHash ()
|
2016-03-23 18:28:03 +03:00
|
|
|
|
-- TODO: another absolute link [absolute-links]
|
2016-05-03 03:27:30 +03:00
|
|
|
|
h1_ (mkLink "The Haskeller's guide" "/haskell")
|
2016-04-06 01:36:55 +03:00
|
|
|
|
renderNoScriptWarning
|
2016-04-21 17:01:54 +03:00
|
|
|
|
renderSearch Nothing
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderCategory category
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderNoScriptWarning :: MonadRandom m => HtmlT m ()
|
2016-04-06 01:36:55 +03:00
|
|
|
|
renderNoScriptWarning =
|
|
|
|
|
noscript_ $ div_ [id_ "noscript-message"] $
|
|
|
|
|
toHtml $ renderMarkdownBlock [text|
|
|
|
|
|
You have Javascript disabled! This site works fine without
|
|
|
|
|
Javascript, but since all editing needs Javascript to work,
|
|
|
|
|
you won't be able to edit anything.
|
|
|
|
|
|]
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderTracking
|
|
|
|
|
:: (MonadIO m, MonadReader Config m) => HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
renderTracking = do
|
2016-03-22 01:17:53 +03:00
|
|
|
|
trackingEnabled <- lift (asks _trackingEnabled)
|
2016-03-16 23:47:59 +03:00
|
|
|
|
when trackingEnabled $ do
|
|
|
|
|
tracking <- liftIO $ T.readFile "static/tracking.html"
|
|
|
|
|
toHtmlRaw tracking
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderDonate
|
|
|
|
|
:: (MonadIO m, MonadRandom m, MonadReader Config m) => HtmlT m ()
|
2016-03-23 18:28:03 +03:00
|
|
|
|
renderDonate = wrapPage "Donate to Artyom" $ do
|
|
|
|
|
toHtmlRaw =<< liftIO (readFile "static/donate.html")
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderUnwrittenRules
|
|
|
|
|
:: (MonadIO m, MonadRandom m, MonadReader Config m) => HtmlT m ()
|
2016-03-20 13:50:23 +03:00
|
|
|
|
renderUnwrittenRules = wrapPage "Unwritten rules" $ do
|
|
|
|
|
toHtml . renderMarkdownBlock =<<
|
|
|
|
|
liftIO (T.readFile "static/unwritten-rules.md")
|
|
|
|
|
|
2016-05-02 21:27:49 +03:00
|
|
|
|
renderMarkdownHelp
|
|
|
|
|
:: (MonadIO m, MonadRandom m, MonadReader Config m) => HtmlT m ()
|
|
|
|
|
renderMarkdownHelp = wrapPage "Markdown" $ do
|
|
|
|
|
toHtml . renderMarkdownBlock =<<
|
|
|
|
|
liftIO (T.readFile "static/markdown.md")
|
|
|
|
|
|
2016-03-19 02:40:00 +03:00
|
|
|
|
-- Include all the necessary things
|
2016-03-22 01:17:53 +03:00
|
|
|
|
wrapPage
|
2016-04-21 16:47:14 +03:00
|
|
|
|
:: (MonadIO m, MonadRandom m, MonadReader Config m)
|
2016-03-22 01:17:53 +03:00
|
|
|
|
=> Text -- ^ Page title
|
|
|
|
|
-> HtmlT m ()
|
|
|
|
|
-> HtmlT m ()
|
2016-03-19 02:40:00 +03:00
|
|
|
|
wrapPage pageTitle page = doctypehtml_ $ do
|
|
|
|
|
head_ $ do
|
|
|
|
|
title_ (toHtml pageTitle)
|
2016-03-19 03:55:31 +03:00
|
|
|
|
meta_ [name_ "viewport",
|
|
|
|
|
content_ "width=device-width, initial-scale=1.0, user-scalable=yes"]
|
2016-03-23 12:22:14 +03:00
|
|
|
|
-- Report all Javascript errors with alerts
|
|
|
|
|
script_ [text|
|
|
|
|
|
window.onerror = function (msg, url, lineNo, columnNo, error) {
|
|
|
|
|
alert("Error in "+url+" at "+lineNo+":"+columnNo+": "+msg+
|
|
|
|
|
"\n\n"+
|
|
|
|
|
"========== Please report it! =========="+
|
|
|
|
|
"\n\n"+
|
|
|
|
|
"https://github.com/aelve/guide/issues");
|
|
|
|
|
return false; };
|
|
|
|
|
|]
|
2016-04-06 01:51:50 +03:00
|
|
|
|
includeJS "/jquery-2.2.0.min.js"
|
2016-03-19 02:40:00 +03:00
|
|
|
|
-- See Note [autosize]
|
2016-04-06 01:51:50 +03:00
|
|
|
|
includeJS "/autosize-3.0.15.min.js"
|
2016-03-19 02:40:00 +03:00
|
|
|
|
onPageLoad (JS "autosize($('textarea'));")
|
2016-04-09 03:36:38 +03:00
|
|
|
|
-- The order is important – markup.css overrides some rules from
|
|
|
|
|
-- highlight.css (e.g. div.sourceCode), css.css overrides the rule for
|
|
|
|
|
-- a.anchor from markup.css.
|
|
|
|
|
--
|
|
|
|
|
-- TODO: maybe use !important or something instead?
|
2016-03-19 02:40:00 +03:00
|
|
|
|
includeCSS "/highlight.css"
|
2016-04-09 03:36:38 +03:00
|
|
|
|
includeCSS "/markup.css"
|
2016-03-19 02:40:00 +03:00
|
|
|
|
includeCSS "/css.css"
|
2016-04-16 23:06:59 +03:00
|
|
|
|
includeCSS "/loader.css"
|
2016-03-19 02:40:00 +03:00
|
|
|
|
-- Include definitions of all Javascript functions that we have defined
|
|
|
|
|
-- in this file. (This isn't an actual file, so don't look for it in the
|
|
|
|
|
-- static folder – it's generated and served in 'otherMethods'.)
|
|
|
|
|
includeJS "/js.js"
|
|
|
|
|
renderTracking
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-- CSS that makes 'shown' and 'noScriptShown' work;
|
|
|
|
|
-- see Note [show-hide]
|
2016-03-19 02:40:00 +03:00
|
|
|
|
noscript_ $ style_ [text|
|
|
|
|
|
.section:not(.noscript-shown) {display:none;}
|
|
|
|
|
|]
|
|
|
|
|
script_ [text|
|
|
|
|
|
var sheet = document.createElement('style');
|
|
|
|
|
sheet.innerHTML = '.section:not(.shown) {display:none;}';
|
|
|
|
|
// “head” instead of “body” because body isn't loaded yet
|
|
|
|
|
document.head.appendChild(sheet);
|
|
|
|
|
|]
|
|
|
|
|
|
|
|
|
|
body_ $ do
|
2016-04-16 23:06:59 +03:00
|
|
|
|
script_ $ fromJS $ JS.createAjaxIndicator ()
|
2016-03-23 18:28:03 +03:00
|
|
|
|
div_ [id_ "main"] $
|
|
|
|
|
page
|
2016-03-19 02:40:00 +03:00
|
|
|
|
div_ [id_ "footer"] $ do
|
2016-05-03 03:27:30 +03:00
|
|
|
|
mapM_ (div_ [class_ "footer-item"]) $
|
|
|
|
|
[ do "made by "
|
|
|
|
|
mkLink "Artyom" "https://artyom.me"
|
|
|
|
|
, do mkLink "source" "https://github.com/aelve/guide"
|
|
|
|
|
"/"
|
|
|
|
|
mkLink "issue tracker" "https://github.com/aelve/guide/issues"
|
|
|
|
|
, mkLink "rules" "/unwritten-rules"
|
|
|
|
|
, do div_ (mkLink "donate" "/donate")
|
|
|
|
|
div_ [class_ "unemployed"] "I'm kinda jobless"
|
|
|
|
|
, do "licensed under "
|
|
|
|
|
mkLink "CC BY-SA 3.0"
|
|
|
|
|
"https://creativecommons.org/licenses/by-sa/3.0/"
|
|
|
|
|
]
|
2016-03-19 02:40:00 +03:00
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- TODO: allow archiving items if they are in every way worse than the rest,
|
|
|
|
|
-- or something (but searching should still be possible)
|
|
|
|
|
|
|
|
|
|
-- TODO: add a list for “interesting libraries, but too lazy to describe, so
|
|
|
|
|
-- somebody describe them for me”
|
|
|
|
|
|
2016-04-21 17:01:54 +03:00
|
|
|
|
renderSearch :: Monad m => Maybe Text -> HtmlT m ()
|
|
|
|
|
renderSearch mbSearchQuery = do
|
|
|
|
|
form_ [action_ "/haskell"] $ do
|
|
|
|
|
input_ [type_ "text", name_ "q", id_ "search", placeholder_ "search",
|
|
|
|
|
value_ (fromMaybe "" mbSearchQuery)]
|
|
|
|
|
|
2016-05-01 16:28:10 +03:00
|
|
|
|
-- If the presentation of the category list ever changes (e.g. to include
|
|
|
|
|
-- lists of items in categories, or their counts, or something), you might
|
|
|
|
|
-- have to start invalidating 'CacheCategoryList' in more things in
|
|
|
|
|
-- 'Cache.invalidateCache'.
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderCategoryList :: (MonadIO m, MonadRandom m) => [Category] -> HtmlT m ()
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderCategoryList cats = cached CacheCategoryList $ do
|
2016-05-01 23:17:55 +03:00
|
|
|
|
div_ [id_ "categories"] $
|
|
|
|
|
for_ (groupWith (view group_) cats) $ \gr ->
|
|
|
|
|
div_ [class_ "category-group"] $ do
|
|
|
|
|
h2_ $ toHtml (gr^?!_head.group_)
|
|
|
|
|
for gr $ \category -> do
|
|
|
|
|
-- TODO: this link shouldn't be absolute [absolute-links]
|
|
|
|
|
a_ [href_ ("/haskell/" <> categorySlug category)] $
|
|
|
|
|
toHtml (category^.title)
|
|
|
|
|
br_ []
|
|
|
|
|
|
|
|
|
|
renderSearchResults :: Monad m => [Category] -> HtmlT m ()
|
|
|
|
|
renderSearchResults cats = do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
div_ [id_ "categories"] $
|
2016-04-21 18:10:58 +03:00
|
|
|
|
for_ cats $ \category -> do
|
|
|
|
|
-- TODO: this link shouldn't be absolute [absolute-links]
|
|
|
|
|
a_ [href_ ("/haskell/" <> categorySlug category)] $
|
|
|
|
|
toHtml (category^.title)
|
|
|
|
|
br_ []
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-05-01 23:17:55 +03:00
|
|
|
|
renderCategoryHeader :: MonadIO m => Category -> HtmlT m ()
|
|
|
|
|
renderCategoryHeader category = cached (CacheCategoryHeader (category^.uid)) $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let thisId = "category-title-" <> uidToText (category^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
2016-03-22 20:19:05 +03:00
|
|
|
|
h2_ [id_ thisId, class_ "category-title"] $ do
|
|
|
|
|
-- TODO: this link shouldn't be absolute [absolute-links]
|
|
|
|
|
span_ [class_ "controls"] $
|
|
|
|
|
a_ [href_ ("/haskell/feed/category/" <> uidToText (category^.uid))] $
|
2016-03-22 20:20:34 +03:00
|
|
|
|
img_ [src_ "/rss-alt.svg",
|
|
|
|
|
alt_ "category feed", title_ "category feed"]
|
2016-03-22 20:19:05 +03:00
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
sectionSpan "normal" [shown, noScriptShown] $ do
|
2016-03-20 02:36:16 +03:00
|
|
|
|
-- TODO: this link shouldn't be absolute [absolute-links]
|
|
|
|
|
a_ [href_ ("/haskell/" <> categorySlug category)] $
|
2016-03-19 02:40:00 +03:00
|
|
|
|
toHtml (category^.title)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
emptySpan "1em"
|
2016-05-01 23:17:55 +03:00
|
|
|
|
span_ [class_ "group"] $
|
|
|
|
|
toHtml (category^.group_)
|
|
|
|
|
emptySpan "1em"
|
2016-03-15 15:35:35 +03:00
|
|
|
|
textButton "edit" $
|
|
|
|
|
JS.switchSection (this, "editing" :: Text)
|
2016-04-07 15:54:11 +03:00
|
|
|
|
emptySpan "1em"
|
|
|
|
|
textButton "delete" $
|
2016-04-22 01:26:45 +03:00
|
|
|
|
JS.deleteCategoryAndRedirect [category^.uid]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
sectionSpan "editing" [] $ do
|
|
|
|
|
textInput [
|
|
|
|
|
value_ (category^.title),
|
|
|
|
|
autocomplete_ "off",
|
|
|
|
|
onEnter $
|
|
|
|
|
JS.submitCategoryTitle (this, category^.uid, inputValue)]
|
2016-05-01 23:17:55 +03:00
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
textInput [
|
|
|
|
|
class_ "group",
|
|
|
|
|
value_ (category^.group_),
|
|
|
|
|
autocomplete_ "off",
|
|
|
|
|
onEnter $
|
|
|
|
|
JS.submitCategoryGroup (this, category^.uid, inputValue)]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
emptySpan "1em"
|
|
|
|
|
textButton "cancel" $
|
|
|
|
|
JS.switchSection (this, "normal" :: Text)
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderCategoryNotes :: (MonadIO m, MonadRandom m) => Category -> HtmlT m ()
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderCategoryNotes category = cached (CacheCategoryNotes (category^.uid)) $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let thisId = "category-notes-" <> uidToText (category^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
2016-04-23 19:30:43 +03:00
|
|
|
|
div_ [id_ thisId, class_ "category-notes"] $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
section "normal" [shown, noScriptShown] $ do
|
2016-04-21 19:28:35 +03:00
|
|
|
|
div_ [class_ "notes-like"] $ do
|
2016-04-22 01:06:02 +03:00
|
|
|
|
if markdownNull (category^.notes)
|
2016-04-21 19:28:35 +03:00
|
|
|
|
then p_ "write something here!"
|
|
|
|
|
else toHtml (category^.notes)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
textButton "edit description" $
|
|
|
|
|
JS.switchSection (this, "editing" :: Text)
|
|
|
|
|
|
2016-04-15 16:15:02 +03:00
|
|
|
|
section "editing" [] $ do
|
2016-04-22 01:06:02 +03:00
|
|
|
|
contents <- if markdownNull (category^.notes)
|
2016-04-15 16:15:02 +03:00
|
|
|
|
then liftIO $ renderMarkdownBlock <$>
|
|
|
|
|
T.readFile "static/category-notes-template.md"
|
|
|
|
|
else return (category^.notes)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
markdownEditor
|
2016-03-17 15:29:45 +03:00
|
|
|
|
[rows_ "10"]
|
2016-04-15 16:15:02 +03:00
|
|
|
|
contents
|
2016-03-15 15:35:35 +03:00
|
|
|
|
(\val -> JS.submitCategoryNotes (this, category^.uid, val))
|
|
|
|
|
(JS.switchSection (this, "normal" :: Text))
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderCategory :: (MonadIO m, MonadRandom m) => Category -> HtmlT m ()
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderCategory category = cached (CacheCategory (category^.uid)) $ do
|
2016-04-07 15:54:11 +03:00
|
|
|
|
div_ [class_ "category", id_ (categoryNodeId category)] $ do
|
2016-05-01 23:17:55 +03:00
|
|
|
|
renderCategoryHeader category
|
2016-03-15 15:35:35 +03:00
|
|
|
|
renderCategoryNotes category
|
|
|
|
|
itemsNode <- div_ [class_ "items"] $ do
|
|
|
|
|
mapM_ (renderItem category) (category^.items)
|
|
|
|
|
thisNode
|
|
|
|
|
textInput [
|
|
|
|
|
placeholder_ "add an item",
|
|
|
|
|
autocomplete_ "off",
|
|
|
|
|
onEnter $ JS.addItem (itemsNode, category^.uid, inputValue) <>
|
|
|
|
|
clearInput ]
|
|
|
|
|
|
|
|
|
|
getItemHue :: Category -> Item -> Hue
|
|
|
|
|
getItemHue category item = case item^.group_ of
|
|
|
|
|
Nothing -> NoHue
|
|
|
|
|
Just s -> M.findWithDefault NoHue s (category^.groups)
|
|
|
|
|
|
|
|
|
|
-- TODO: perhaps use jQuery Touch Punch or something to allow dragging items
|
|
|
|
|
-- instead of using arrows? Touch Punch works on mobile, too
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderItem :: (MonadIO m, MonadRandom m) => Category -> Item -> HtmlT m ()
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderItem category item = cached (CacheItem (item^.uid)) $ do
|
2016-03-20 13:30:10 +03:00
|
|
|
|
-- The id is used for links in feeds, and for anchor links
|
2016-04-07 15:54:11 +03:00
|
|
|
|
div_ [id_ (itemNodeId item), class_ "item"] $ do
|
2016-03-17 03:35:56 +03:00
|
|
|
|
renderItemInfo category item
|
|
|
|
|
let bg = hueToLightColor $ getItemHue category item
|
|
|
|
|
div_ [class_ "item-body", style_ ("background-color:" <> bg)] $ do
|
|
|
|
|
renderItemDescription item
|
|
|
|
|
renderItemTraits item
|
2016-03-17 14:58:44 +03:00
|
|
|
|
renderItemEcosystem item
|
2016-04-22 01:06:02 +03:00
|
|
|
|
renderItemNotes category item
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
-- TODO: warn when a library isn't on Hackage but is supposed to be
|
2016-03-17 03:35:56 +03:00
|
|
|
|
|
2016-03-21 02:12:03 +03:00
|
|
|
|
renderItemTitle :: Monad m => Item -> HtmlT m ()
|
|
|
|
|
renderItemTitle item = do
|
2016-05-02 16:54:35 +03:00
|
|
|
|
case item^.link of
|
|
|
|
|
Just l -> a_ [href_ l] (toHtml (item^.name))
|
|
|
|
|
Nothing -> toHtml (item^.name)
|
2016-03-19 21:36:21 +03:00
|
|
|
|
let hackageLink x = "https://hackage.haskell.org/package/" <> x
|
2016-05-02 16:54:35 +03:00
|
|
|
|
case item ^. kind.hackageName of
|
|
|
|
|
Just x -> " (" >> a_ [href_ (hackageLink x)] "Hackage" >> ")"
|
|
|
|
|
Nothing -> return ()
|
2016-03-19 21:36:21 +03:00
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- TODO: give a link to oldest available docs when the new docs aren't there
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderItemInfo :: (MonadIO m, MonadRandom m) => Category -> Item -> HtmlT m ()
|
|
|
|
|
renderItemInfo cat item = cached (CacheItemInfo (item^.uid)) $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let bg = hueToDarkColor $ getItemHue cat item
|
|
|
|
|
let thisId = "item-info-" <> uidToText (item^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
|
|
|
|
div_ [id_ thisId, class_ "item-info",
|
|
|
|
|
style_ ("background-color:" <> bg)] $ do
|
|
|
|
|
|
|
|
|
|
section "normal" [shown, noScriptShown] $ do
|
|
|
|
|
-- TODO: [very-easy] move this style_ into css.css
|
|
|
|
|
span_ [style_ "font-size:150%"] $ do
|
2016-03-20 13:30:10 +03:00
|
|
|
|
-- TODO: absolute links again [absolute-links]
|
2016-04-22 01:06:02 +03:00
|
|
|
|
a_ [class_ "anchor", href_ (itemLink cat item)] "#"
|
2016-03-21 02:12:03 +03:00
|
|
|
|
renderItemTitle item
|
2016-03-15 15:35:35 +03:00
|
|
|
|
emptySpan "2em"
|
|
|
|
|
toHtml (fromMaybe "other" (item^.group_))
|
|
|
|
|
span_ [class_ "controls"] $ do
|
|
|
|
|
imgButton "move item up" "/arrow-thick-top.svg" [] $
|
2016-04-07 15:54:11 +03:00
|
|
|
|
JS.moveItemUp (item^.uid, itemNode item)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
imgButton "move item down" "/arrow-thick-bottom.svg" [] $
|
2016-04-07 15:54:11 +03:00
|
|
|
|
JS.moveItemDown (item^.uid, itemNode item)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
emptySpan "1.5em"
|
|
|
|
|
imgButton "edit item info" "/pencil.svg" [] $
|
|
|
|
|
JS.switchSection (this, "editing" :: Text)
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
imgButton "delete item" "/x.svg" [] $
|
2016-04-07 15:54:11 +03:00
|
|
|
|
JS.deleteItem (item^.uid, itemNode item)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- TODO: link to Stackage too
|
|
|
|
|
-- TODO: should check for Stackage automatically
|
|
|
|
|
|
|
|
|
|
section "editing" [] $ do
|
|
|
|
|
let selectedIf p x = if p then with x [selected_ "selected"] else x
|
2016-03-17 03:35:56 +03:00
|
|
|
|
-- When the info/header node changes its group (and is hence
|
|
|
|
|
-- recolored), item's body has to be recolored too
|
|
|
|
|
let bodyNode = JS.selectChildren (JS.selectParent this)
|
|
|
|
|
(JS.selectClass "item-body")
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let formSubmitHandler formNode =
|
2016-03-17 03:35:56 +03:00
|
|
|
|
JS.submitItemInfo (this, bodyNode, item^.uid, formNode)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
form_ [onFormSubmit formSubmitHandler] $ do
|
|
|
|
|
-- All inputs have "autocomplete = off" thanks to
|
|
|
|
|
-- <http://stackoverflow.com/q/8311455>
|
|
|
|
|
label_ $ do
|
|
|
|
|
"Name" >> br_ []
|
|
|
|
|
input_ [type_ "text", name_ "name",
|
|
|
|
|
autocomplete_ "off",
|
|
|
|
|
value_ (item^.name)]
|
|
|
|
|
br_ []
|
|
|
|
|
label_ $ do
|
|
|
|
|
"Kind" >> br_ []
|
|
|
|
|
select_ [name_ "kind"] $ do
|
|
|
|
|
option_ [value_ "library"] "Library"
|
|
|
|
|
& selectedIf (case item^.kind of Library{} -> True; _ -> False)
|
|
|
|
|
option_ [value_ "tool"] "Tool"
|
|
|
|
|
& selectedIf (case item^.kind of Tool{} -> True; _ -> False)
|
|
|
|
|
option_ [value_ "other"] "Other"
|
|
|
|
|
& selectedIf (case item^.kind of Other{} -> True; _ -> False)
|
|
|
|
|
br_ []
|
|
|
|
|
label_ $ do
|
|
|
|
|
"Name on Hackage" >> br_ []
|
|
|
|
|
input_ [type_ "text", name_ "hackage-name", autocomplete_ "off",
|
|
|
|
|
value_ (fromMaybe "" (item^?kind.hackageName._Just))]
|
|
|
|
|
br_ []
|
|
|
|
|
label_ $ do
|
|
|
|
|
"Site (optional)" >> br_ []
|
|
|
|
|
input_ [type_ "text", name_ "link", autocomplete_ "off",
|
|
|
|
|
value_ (fromMaybe "" (item^.link))]
|
|
|
|
|
br_ []
|
2016-03-19 00:05:01 +03:00
|
|
|
|
newGroupInputId <- randomLongUid
|
2016-03-15 15:35:35 +03:00
|
|
|
|
label_ $ do
|
|
|
|
|
"Group" >> br_ []
|
|
|
|
|
-- When “new group” is selected in the list, we show a field for
|
|
|
|
|
-- entering new group's name
|
|
|
|
|
let selectHandler = [text|
|
|
|
|
|
if (this.value == "$newGroupValue") {
|
|
|
|
|
$("#$idText").show();
|
|
|
|
|
$("#$idText").focus(); }
|
|
|
|
|
else $("#$idText").hide(); |]
|
|
|
|
|
where idText = uidToText newGroupInputId
|
|
|
|
|
select_ [name_ "group", autocomplete_ "off",
|
|
|
|
|
onchange_ selectHandler] $ do
|
|
|
|
|
let gs = Nothing : map Just (M.keys (cat^.groups))
|
|
|
|
|
for_ gs $ \group' -> do
|
|
|
|
|
-- Text that will be shown in the list (“-” stands for “no
|
|
|
|
|
-- group”)
|
|
|
|
|
let txt = fromMaybe "-" group'
|
|
|
|
|
-- If the element corresponds to the current group of the
|
|
|
|
|
-- item (or the element is “-”, i.e. Nothing, and the group
|
|
|
|
|
-- is Nothing too), mark it as selected, thus making it the
|
|
|
|
|
-- element that will be chosen by default when the form is
|
|
|
|
|
-- rendered
|
|
|
|
|
option_ [value_ txt] (toHtml txt)
|
|
|
|
|
& selectedIf (group' == item^.group_)
|
|
|
|
|
option_ [value_ newGroupValue] "New group..."
|
|
|
|
|
input_ [uid_ newGroupInputId, type_ "text", autocomplete_ "off",
|
|
|
|
|
name_ "custom-group", hidden_ "hidden"]
|
|
|
|
|
br_ []
|
|
|
|
|
input_ [type_ "submit", value_ "Save"]
|
|
|
|
|
button "Cancel" [] $
|
|
|
|
|
JS.switchSection (this, "normal" :: Text)
|
|
|
|
|
|
|
|
|
|
-- TODO: categories that don't directly compare libraries but just list all
|
|
|
|
|
-- libraries about something (e.g. Yesod plugins, or whatever)
|
|
|
|
|
|
2016-05-01 16:28:10 +03:00
|
|
|
|
-- TODO: just make a synonym for “Html with IO and randomness”
|
|
|
|
|
|
|
|
|
|
renderItemDescription :: (MonadIO m, MonadRandom m) => Item -> HtmlT m ()
|
|
|
|
|
renderItemDescription item = cached (CacheItemDescription (item^.uid)) $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let thisId = "item-description-" <> uidToText (item^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
2016-03-17 03:35:56 +03:00
|
|
|
|
div_ [id_ thisId, class_ "item-description"] $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
section "normal" [shown, noScriptShown] $ do
|
2016-04-22 16:29:18 +03:00
|
|
|
|
strong_ "Summary"
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
imgButton "edit summary" "/pencil.svg"
|
|
|
|
|
[style_ "width:12px;opacity:0.5"] $
|
|
|
|
|
JS.switchSection (this, "editing" :: Text)
|
2016-04-21 19:28:35 +03:00
|
|
|
|
div_ [class_ "notes-like"] $ do
|
2016-04-22 01:06:02 +03:00
|
|
|
|
if markdownNull (item^.description)
|
2016-04-21 19:28:35 +03:00
|
|
|
|
then p_ "write something here!"
|
|
|
|
|
else toHtml (item^.description)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-04-22 16:29:18 +03:00
|
|
|
|
section "editing" [] $ do
|
|
|
|
|
strong_ "Summary"
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
imgButton "undo editing summary" "/pencil.svg"
|
|
|
|
|
[style_ "width:12px;opacity:0.5"] $
|
|
|
|
|
JS.switchSection (this, "normal" :: Text)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
markdownEditor
|
2016-03-17 15:29:45 +03:00
|
|
|
|
[rows_ "10"]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
(item^.description)
|
|
|
|
|
(\val -> JS.submitItemDescription (this, item^.uid, val))
|
|
|
|
|
(JS.switchSection (this, "normal" :: Text))
|
|
|
|
|
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderItemEcosystem :: (MonadIO m, MonadRandom m) => Item -> HtmlT m ()
|
|
|
|
|
renderItemEcosystem item = cached (CacheItemEcosystem (item^.uid)) $ do
|
2016-03-17 02:52:40 +03:00
|
|
|
|
let thisId = "item-ecosystem-" <> uidToText (item^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
2016-03-17 03:35:56 +03:00
|
|
|
|
div_ [id_ thisId, class_ "item-ecosystem"] $ do
|
2016-03-17 02:52:40 +03:00
|
|
|
|
|
|
|
|
|
section "normal" [shown, noScriptShown] $ do
|
2016-04-22 16:22:46 +03:00
|
|
|
|
strong_ "Ecosystem"
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
imgButton "edit ecosystem" "/pencil.svg"
|
|
|
|
|
[style_ "width:12px;opacity:0.5"] $
|
|
|
|
|
JS.switchSection (this, "editing" :: Text)
|
2016-04-22 01:06:02 +03:00
|
|
|
|
unless (markdownNull (item^.ecosystem)) $
|
2016-03-17 15:29:45 +03:00
|
|
|
|
toHtml (item^.ecosystem)
|
2016-03-17 02:52:40 +03:00
|
|
|
|
|
2016-04-22 16:22:46 +03:00
|
|
|
|
section "editing" [] $ do
|
|
|
|
|
strong_ "Ecosystem"
|
|
|
|
|
emptySpan "0.5em"
|
|
|
|
|
imgButton "undo editing ecosystem" "/pencil.svg"
|
|
|
|
|
[style_ "width:12px;opacity:0.5"] $
|
|
|
|
|
JS.switchSection (this, "normal" :: Text)
|
2016-03-17 15:29:45 +03:00
|
|
|
|
markdownEditor
|
2016-03-17 02:52:40 +03:00
|
|
|
|
[rows_ "3"]
|
|
|
|
|
(item^.ecosystem)
|
|
|
|
|
(\val -> JS.submitItemEcosystem (this, item^.uid, val))
|
2016-03-17 15:29:45 +03:00
|
|
|
|
(JS.switchSection (this, "normal" :: Text))
|
2016-03-17 02:52:40 +03:00
|
|
|
|
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderItemTraits :: (MonadIO m, MonadRandom m) => Item -> HtmlT m ()
|
|
|
|
|
renderItemTraits item = cached (CacheItemTraits (item^.uid)) $ do
|
2016-03-17 03:35:56 +03:00
|
|
|
|
div_ [class_ "item-traits"] $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
this <- thisNode
|
|
|
|
|
div_ [class_ "traits-groups-container"] $ do
|
|
|
|
|
div_ [class_ "traits-group"] $ do
|
2016-03-17 03:02:51 +03:00
|
|
|
|
strong_ "Pros"
|
2016-03-16 15:46:29 +03:00
|
|
|
|
-- We can't use 'thisNode' inside <ul> because it creates a <span>
|
|
|
|
|
-- and only <li> elements can be children of <ul>
|
2016-03-19 00:05:01 +03:00
|
|
|
|
listUid <- randomLongUid
|
2016-03-16 15:46:29 +03:00
|
|
|
|
ul_ [uid_ listUid] $
|
2016-03-15 15:35:35 +03:00
|
|
|
|
mapM_ (renderTrait (item^.uid)) (item^.pros)
|
|
|
|
|
section "editable" [] $
|
|
|
|
|
smallMarkdownEditor
|
|
|
|
|
[rows_ "3", placeholder_ "add pro"]
|
2016-04-22 01:06:02 +03:00
|
|
|
|
(renderMarkdownInline "")
|
2016-03-16 15:46:29 +03:00
|
|
|
|
(\val -> JS.addPro (JS.selectUid listUid, item^.uid, val) <>
|
2016-03-15 15:35:35 +03:00
|
|
|
|
JS.assign val ("" :: Text))
|
|
|
|
|
Nothing
|
|
|
|
|
div_ [class_ "traits-group"] $ do
|
2016-03-17 03:02:51 +03:00
|
|
|
|
strong_ "Cons"
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- TODO: [easy] maybe add a line here?
|
2016-03-19 00:05:01 +03:00
|
|
|
|
listUid <- randomLongUid
|
2016-03-16 15:46:29 +03:00
|
|
|
|
ul_ [uid_ listUid] $
|
2016-03-15 15:35:35 +03:00
|
|
|
|
mapM_ (renderTrait (item^.uid)) (item^.cons)
|
|
|
|
|
section "editable" [] $
|
|
|
|
|
smallMarkdownEditor
|
|
|
|
|
[rows_ "3", placeholder_ "add con"]
|
2016-04-22 01:06:02 +03:00
|
|
|
|
(renderMarkdownInline "")
|
2016-03-16 15:46:29 +03:00
|
|
|
|
(\val -> JS.addCon (JS.selectUid listUid, item^.uid, val) <>
|
2016-03-15 15:35:35 +03:00
|
|
|
|
JS.assign val ("" :: Text))
|
|
|
|
|
Nothing
|
|
|
|
|
section "normal" [shown, noScriptShown] $ do
|
|
|
|
|
textButton "edit pros/cons" $
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-- Switches sections in *all* traits
|
2016-03-15 15:35:35 +03:00
|
|
|
|
JS.switchSectionsEverywhere(this, "editable" :: Text)
|
|
|
|
|
section "editable" [] $ do
|
|
|
|
|
textButton "edit off" $
|
|
|
|
|
JS.switchSectionsEverywhere(this, "normal" :: Text)
|
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
renderTrait :: MonadRandom m => Uid Item -> Trait -> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
renderTrait itemId trait = do
|
|
|
|
|
let thisId = "trait-" <> uidToText (trait^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
2016-04-19 20:03:54 +03:00
|
|
|
|
editingSectionUid <- randomLongUid
|
2016-03-15 15:35:35 +03:00
|
|
|
|
li_ [id_ thisId] $ do
|
|
|
|
|
|
2016-04-11 23:31:30 +03:00
|
|
|
|
sectionSpan "normal editable" [shown, noScriptShown] $ do
|
2016-03-16 02:17:08 +03:00
|
|
|
|
toHtml (trait^.content)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-04-11 23:31:30 +03:00
|
|
|
|
sectionSpan "editable" [] $ do
|
2016-03-15 15:35:35 +03:00
|
|
|
|
br_ []
|
|
|
|
|
imgButton "move trait up" "/arrow-thick-top.svg" [width_ "12"] $
|
|
|
|
|
JS.moveTraitUp (itemId, trait^.uid, this)
|
|
|
|
|
imgButton "move trait down" "/arrow-thick-bottom.svg" [width_ "12"] $
|
|
|
|
|
JS.moveTraitDown (itemId, trait^.uid, this)
|
|
|
|
|
-- TODO: these 3 icons in a row don't look nice
|
|
|
|
|
imgButton "delete trait" "/x.svg" [width_ "12"] $
|
|
|
|
|
JS.deleteTrait (itemId, trait^.uid, this)
|
2016-04-19 20:03:54 +03:00
|
|
|
|
textareaUid <- randomLongUid
|
2016-03-15 15:35:35 +03:00
|
|
|
|
textButton "edit" $
|
2016-04-22 16:59:50 +03:00
|
|
|
|
-- See Note [dynamic interface]
|
2016-04-19 20:03:54 +03:00
|
|
|
|
JS.makeTraitEditor (this, JS.selectUid editingSectionUid,
|
|
|
|
|
textareaUid,
|
|
|
|
|
trait^.content.mdText,
|
|
|
|
|
itemId, trait^.uid) <>
|
|
|
|
|
JS.switchSection (this, "editing" :: Text) <>
|
|
|
|
|
JS.autosizeTextarea [JS.selectUid textareaUid]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-04-19 20:03:54 +03:00
|
|
|
|
section "editing" [uid_ editingSectionUid] $ do
|
|
|
|
|
return ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
-- TODO: automatically provide links to modules in Markdown (and have a
|
|
|
|
|
-- database of modules or something)
|
|
|
|
|
|
|
|
|
|
-- TODO: [very-easy] write about the all-is-text extension
|
|
|
|
|
-- TODO: [easy] write that arrows are for arranging stuff, not upvoting
|
|
|
|
|
|
|
|
|
|
-- TODO: [easy] add Hayoo search, Hoogle search, and Hackage search shortcut
|
|
|
|
|
-- boxes
|
|
|
|
|
|
|
|
|
|
-- TODO: when searching, show links to package, main modules, etc before all
|
|
|
|
|
-- categories
|
|
|
|
|
|
|
|
|
|
-- TODO: attach TODOs (“fix grammar”, etc) to items and categories (or should
|
|
|
|
|
-- people instead just write “TODO fix grammar” in description and then such
|
|
|
|
|
-- things could be displayed in gray font and also there'd be an
|
|
|
|
|
-- automatically updated list of TODOs somewhere?)
|
|
|
|
|
|
|
|
|
|
-- TODO: [very-easy] focus the notes textarea on edit (can use jQuery's
|
|
|
|
|
-- .focus() on it)
|
2016-04-22 01:06:02 +03:00
|
|
|
|
renderItemNotes
|
|
|
|
|
:: (MonadIO m, MonadRandom m)
|
|
|
|
|
=> Category -> Item -> HtmlT m ()
|
2016-05-01 16:28:10 +03:00
|
|
|
|
renderItemNotes category item = cached (CacheItemNotes (item^.uid)) $ do
|
2016-04-20 01:59:29 +03:00
|
|
|
|
-- Don't change this ID, it's used in e.g. 'JS.expandHash'
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let thisId = "item-notes-" <> uidToText (item^.uid)
|
|
|
|
|
this = JS.selectId thisId
|
2016-04-19 20:03:54 +03:00
|
|
|
|
editingSectionUid <- randomLongUid
|
2016-03-17 03:35:56 +03:00
|
|
|
|
div_ [id_ thisId, class_ "item-notes"] $ do
|
2016-04-22 16:18:58 +03:00
|
|
|
|
let notesLink = format "/haskell/{}#{}"
|
|
|
|
|
(categorySlug category, thisId)
|
|
|
|
|
a_ [href_ notesLink] $
|
|
|
|
|
strong_ "Notes"
|
|
|
|
|
|
|
|
|
|
let renderTree :: Monad m => Forest (Inlines, Text) -> HtmlT m ()
|
|
|
|
|
renderTree [] = return ()
|
|
|
|
|
renderTree xs = ul_ $ do
|
|
|
|
|
for_ xs $ \(Node (is, id') children) -> li_ $ do
|
|
|
|
|
let handler = fromJS (JS.expandItemNotes [item^.uid])
|
|
|
|
|
-- The link has to be full because sometimes we are
|
|
|
|
|
-- looking at items from pages different from the
|
|
|
|
|
-- proper category pages (e.g. if a search returned a
|
|
|
|
|
-- list of items). Well, actually it doesn't happen
|
|
|
|
|
-- yet (at the moment of writing), but it might start
|
|
|
|
|
-- happening and then it's better to be prepared.
|
|
|
|
|
fullLink = format "/haskell/{}#{}"
|
|
|
|
|
(categorySlug category, id')
|
|
|
|
|
a_ [href_ fullLink, onclick_ handler] $
|
|
|
|
|
renderInlines def is
|
|
|
|
|
renderTree children
|
2016-04-21 19:40:00 +03:00
|
|
|
|
let renderTOC = do
|
2016-04-22 01:06:02 +03:00
|
|
|
|
let toc = item^.notes.mdTOC
|
2016-04-21 19:40:00 +03:00
|
|
|
|
div_ [class_ "notes-toc"] $ do
|
|
|
|
|
if null toc
|
2016-04-22 01:06:02 +03:00
|
|
|
|
then p_ (emptySpan "1.5em" >> "<notes are empty>")
|
2016-04-21 19:40:00 +03:00
|
|
|
|
else renderTree toc
|
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
section "collapsed" [shown] $ do
|
2016-04-16 02:02:43 +03:00
|
|
|
|
textButton "expand notes" $
|
2016-04-20 01:59:29 +03:00
|
|
|
|
JS.expandItemNotes [item^.uid]
|
2016-04-21 19:40:00 +03:00
|
|
|
|
renderTOC
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
section "expanded" [noScriptShown] $ do
|
2016-04-19 20:03:54 +03:00
|
|
|
|
textareaUid <- randomLongUid
|
2016-04-22 01:06:02 +03:00
|
|
|
|
contents <- if markdownNull (item^.notes)
|
2016-04-19 20:03:54 +03:00
|
|
|
|
then liftIO $ T.readFile "static/item-notes-template.md"
|
|
|
|
|
else return (item^.notes.mdText)
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let buttons = do
|
2016-04-16 02:02:43 +03:00
|
|
|
|
textButton "collapse notes" $
|
2016-03-15 15:35:35 +03:00
|
|
|
|
JS.switchSection (this, "collapsed" :: Text)
|
2016-03-15 15:41:09 +03:00
|
|
|
|
emptySpan "1em"
|
|
|
|
|
textButton "edit notes" $
|
2016-04-22 16:59:50 +03:00
|
|
|
|
-- See Note [dynamic interface]
|
2016-04-19 20:03:54 +03:00
|
|
|
|
JS.makeItemNotesEditor (
|
|
|
|
|
this, JS.selectUid editingSectionUid,
|
|
|
|
|
textareaUid,
|
|
|
|
|
contents,
|
|
|
|
|
item^.uid) <>
|
|
|
|
|
JS.switchSection (this, "editing" :: Text) <>
|
|
|
|
|
JS.autosizeTextarea [JS.selectUid textareaUid]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
buttons
|
2016-04-21 19:40:00 +03:00
|
|
|
|
renderTOC
|
2016-04-21 19:28:35 +03:00
|
|
|
|
div_ [class_ "notes-like"] $ do
|
2016-04-22 01:06:02 +03:00
|
|
|
|
if markdownNull (item^.notes)
|
2016-04-21 19:28:35 +03:00
|
|
|
|
then p_ "add something!"
|
|
|
|
|
else toHtml (item^.notes)
|
2016-04-22 01:06:02 +03:00
|
|
|
|
unless (markdownNull (item^.notes)) $
|
2016-04-21 19:28:35 +03:00
|
|
|
|
buttons
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- TODO: [easy] the lower “hide notes” should scroll back to item when
|
|
|
|
|
-- the notes are closed (but don't scroll if it's already visible after
|
|
|
|
|
-- the notes have been hidden)
|
|
|
|
|
|
2016-04-19 20:03:54 +03:00
|
|
|
|
section "editing" [uid_ editingSectionUid] $
|
|
|
|
|
return ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
-- TODO: a shortcut for editing (when you press Ctrl-something, whatever was
|
|
|
|
|
-- selected becomes editable)
|
|
|
|
|
|
2016-03-22 01:17:53 +03:00
|
|
|
|
renderItemForFeed :: Monad m => Item -> HtmlT m ()
|
2016-03-19 21:36:21 +03:00
|
|
|
|
renderItemForFeed item = do
|
2016-03-21 02:12:03 +03:00
|
|
|
|
h1_ $ renderItemTitle item
|
2016-04-22 01:06:02 +03:00
|
|
|
|
unless (markdownNull (item^.description)) $
|
2016-03-19 21:36:21 +03:00
|
|
|
|
toHtml (item^.description)
|
|
|
|
|
h2_ "Pros"
|
|
|
|
|
ul_ $ mapM_ (p_ . li_ . toHtml . view content) (item^.pros)
|
|
|
|
|
h2_ "Cons"
|
|
|
|
|
ul_ $ mapM_ (p_ . li_ . toHtml . view content) (item^.cons)
|
2016-04-22 01:06:02 +03:00
|
|
|
|
unless (markdownNull (item^.ecosystem)) $ do
|
2016-03-19 21:36:21 +03:00
|
|
|
|
h2_ "Ecosystem"
|
|
|
|
|
toHtml (item^.ecosystem)
|
2016-04-22 16:18:58 +03:00
|
|
|
|
-- TODO: include .notes-like style here? otherwise the headers are too big
|
2016-04-22 01:06:02 +03:00
|
|
|
|
unless (markdownNull (item^.notes)) $ do
|
2016-03-19 21:36:21 +03:00
|
|
|
|
h2_ "Notes"
|
|
|
|
|
toHtml (item^.notes)
|
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- Utils
|
|
|
|
|
|
2016-03-22 01:17:53 +03:00
|
|
|
|
onPageLoad :: Monad m => JS -> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
onPageLoad js = script_ $ format "$(document).ready(function(){{}});" [js]
|
|
|
|
|
|
2016-03-22 01:17:53 +03:00
|
|
|
|
emptySpan :: Monad m => Text -> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
emptySpan w = span_ [style_ ("margin-left:" <> w)] mempty
|
|
|
|
|
|
|
|
|
|
-- Use inputValue to get the value (works with input_ and textarea_)
|
|
|
|
|
onEnter :: JS -> Attribute
|
|
|
|
|
onEnter handler = onkeydown_ $
|
|
|
|
|
format "if (event.keyCode == 13) {{} return false;}" [handler]
|
|
|
|
|
|
2016-03-22 01:17:53 +03:00
|
|
|
|
textInput :: Monad m => [Attribute] -> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
textInput attrs = input_ (type_ "text" : attrs)
|
|
|
|
|
|
|
|
|
|
inputValue :: JS
|
|
|
|
|
inputValue = JS "this.value"
|
|
|
|
|
|
|
|
|
|
clearInput :: JS
|
|
|
|
|
clearInput = JS "this.value = '';"
|
|
|
|
|
|
|
|
|
|
onFormSubmit :: (JS -> JS) -> Attribute
|
|
|
|
|
onFormSubmit f = onsubmit_ $ format "{} return false;" [f (JS "this")]
|
|
|
|
|
|
2016-03-22 01:17:53 +03:00
|
|
|
|
button :: Monad m => Text -> [Attribute] -> JS -> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
button value attrs handler =
|
|
|
|
|
input_ (type_ "button" : value_ value : onclick_ handler' : attrs)
|
|
|
|
|
where
|
|
|
|
|
handler' = fromJS handler
|
|
|
|
|
|
|
|
|
|
-- A text button looks like “[cancel]”
|
|
|
|
|
textButton
|
2016-03-22 01:17:53 +03:00
|
|
|
|
:: Monad m
|
|
|
|
|
=> Text -- ^ Button text
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-> JS -- ^ Onclick handler
|
2016-03-22 01:17:53 +03:00
|
|
|
|
-> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
textButton caption (JS handler) =
|
|
|
|
|
span_ [class_ "text-button"] $
|
|
|
|
|
-- “#” is used instead of javascript:void(0) because the latter is slow
|
|
|
|
|
-- in Firefox (at least for me – tested with Firefox 43 on Arch Linux)
|
|
|
|
|
a_ [href_ "#", onclick_ (handler <> "return false;")]
|
|
|
|
|
(toHtml caption)
|
|
|
|
|
|
|
|
|
|
-- So far all icons used here have been from <https://useiconic.com/open/>
|
2016-03-22 01:17:53 +03:00
|
|
|
|
imgButton :: Monad m => Text -> Url -> [Attribute] -> JS -> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
imgButton alt src attrs (JS handler) =
|
|
|
|
|
a_ [href_ "#", onclick_ (handler <> "return false;")]
|
2016-03-22 20:20:34 +03:00
|
|
|
|
(img_ (src_ src : alt_ alt : title_ alt : attrs))
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-05-03 03:27:30 +03:00
|
|
|
|
mkLink :: Monad m => HtmlT m a -> Url -> HtmlT m a
|
|
|
|
|
mkLink x src = a_ [href_ src] x
|
|
|
|
|
|
2016-03-15 15:35:35 +03:00
|
|
|
|
markdownEditor
|
2016-04-21 16:47:14 +03:00
|
|
|
|
:: MonadRandom m
|
2016-03-22 01:17:53 +03:00
|
|
|
|
=> [Attribute]
|
2016-03-17 15:29:45 +03:00
|
|
|
|
-> MarkdownBlock -- ^ Default text
|
2016-03-16 02:17:08 +03:00
|
|
|
|
-> (JS -> JS) -- ^ “Submit” handler, receiving the contents of the editor
|
|
|
|
|
-> JS -- ^ “Cancel” handler
|
2016-03-22 01:17:53 +03:00
|
|
|
|
-> HtmlT m ()
|
2016-04-16 02:02:43 +03:00
|
|
|
|
markdownEditor attr (view mdText -> s) submit cancel = do
|
2016-03-24 03:02:09 +03:00
|
|
|
|
textareaUid <- randomLongUid
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- Autocomplete has to be turned off thanks to
|
|
|
|
|
-- <http://stackoverflow.com/q/8311455>.
|
2016-03-24 03:02:09 +03:00
|
|
|
|
textarea_ ([uid_ textareaUid, autocomplete_ "off", class_ "big fullwidth"]
|
2016-03-17 15:29:45 +03:00
|
|
|
|
++ attr) $
|
2016-03-15 15:35:35 +03:00
|
|
|
|
toHtml s
|
2016-03-24 03:02:09 +03:00
|
|
|
|
let val = JS $ format "document.getElementById(\"{}\").value" [textareaUid]
|
2016-03-15 15:35:35 +03:00
|
|
|
|
button "Save" [] $
|
|
|
|
|
submit val
|
|
|
|
|
emptySpan "6px"
|
|
|
|
|
button "Cancel" [] $
|
|
|
|
|
JS.assign val s <>
|
|
|
|
|
cancel
|
|
|
|
|
emptySpan "6px"
|
2016-05-02 21:27:49 +03:00
|
|
|
|
a_ [href_ "/markdown", target_ "_blank"] "Markdown"
|
2016-03-24 03:02:09 +03:00
|
|
|
|
emptySpan "6px"
|
|
|
|
|
-- TODO: this jumps around when there's a lot of text, need to somehow
|
2016-04-19 20:03:54 +03:00
|
|
|
|
-- prevent jumping (and in JS.makeItemNotesEditor too)
|
2016-03-24 03:02:09 +03:00
|
|
|
|
let checkHandler = fromJS $
|
|
|
|
|
JS.setMonospace (JS.selectUid textareaUid, JS "this.checked")
|
|
|
|
|
label_ $ do
|
|
|
|
|
input_ [type_ "checkbox", name_ "monospace", onchange_ checkHandler]
|
|
|
|
|
"monospace editor"
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
|
|
|
|
smallMarkdownEditor
|
2016-04-21 16:47:14 +03:00
|
|
|
|
:: MonadRandom m
|
2016-03-22 01:17:53 +03:00
|
|
|
|
=> [Attribute]
|
2016-03-16 02:17:08 +03:00
|
|
|
|
-> MarkdownInline -- ^ Default text
|
|
|
|
|
-> (JS -> JS) -- ^ “Submit” handler, receiving the contents of the editor
|
|
|
|
|
-> Maybe JS -- ^ “Cancel” handler (if “Cancel” is needed)
|
2016-03-22 01:17:53 +03:00
|
|
|
|
-> HtmlT m ()
|
2016-04-16 02:02:43 +03:00
|
|
|
|
smallMarkdownEditor attr (view mdText -> s) submit mbCancel = do
|
2016-03-19 00:05:01 +03:00
|
|
|
|
textareaId <- randomLongUid
|
2016-03-15 15:35:35 +03:00
|
|
|
|
let val = JS $ format "document.getElementById(\"{}\").value" [textareaId]
|
|
|
|
|
textarea_ ([class_ "fullwidth", uid_ textareaId, autocomplete_ "off",
|
2016-03-17 15:29:45 +03:00
|
|
|
|
onEnter (submit val)] ++ attr) $
|
2016-03-15 15:35:35 +03:00
|
|
|
|
toHtml s
|
|
|
|
|
case mbCancel of
|
|
|
|
|
Nothing -> return ()
|
|
|
|
|
Just cancel -> do
|
|
|
|
|
br_ []
|
|
|
|
|
textButton "cancel" $
|
|
|
|
|
JS.assign val s <>
|
|
|
|
|
cancel
|
2016-05-02 21:27:49 +03:00
|
|
|
|
a_ [href_ "/markdown", target_ "_blank", style_ "float:right"] "Markdown"
|
2016-03-15 15:35:35 +03:00
|
|
|
|
|
2016-04-21 16:47:14 +03:00
|
|
|
|
thisNode :: MonadRandom m => HtmlT m JQuerySelector
|
2016-03-15 15:35:35 +03:00
|
|
|
|
thisNode = do
|
2016-03-19 00:05:01 +03:00
|
|
|
|
uid' <- randomLongUid
|
2016-03-15 15:35:35 +03:00
|
|
|
|
-- If the class name ever changes, fix 'JS.moveNodeUp' and
|
|
|
|
|
-- 'JS.moveNodeDown'.
|
|
|
|
|
span_ [uid_ uid', class_ "dummy"] mempty
|
|
|
|
|
return (JS.selectParent (JS.selectUid uid'))
|
|
|
|
|
|
2016-04-07 15:54:11 +03:00
|
|
|
|
itemNodeId :: Item -> Text
|
|
|
|
|
itemNodeId item = "item-" <> uidToText (item^.uid)
|
|
|
|
|
|
|
|
|
|
itemNode :: Item -> JQuerySelector
|
|
|
|
|
itemNode = JS.selectId . itemNodeId
|
|
|
|
|
|
|
|
|
|
categoryNodeId :: Category -> Text
|
|
|
|
|
categoryNodeId category = "category-" <> uidToText (category^.uid)
|
|
|
|
|
|
2016-04-22 01:06:02 +03:00
|
|
|
|
itemLink :: Category -> Item -> Text
|
|
|
|
|
itemLink category item =
|
|
|
|
|
format "/haskell/{}#{}" (categorySlug category, itemNodeId item)
|
|
|
|
|
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-- See Note [show-hide]; wheh changing these, also look at 'JS.switchSection'.
|
2016-03-15 15:35:35 +03:00
|
|
|
|
shown, noScriptShown :: Attribute
|
|
|
|
|
shown = class_ " shown "
|
|
|
|
|
noScriptShown = class_ " noscript-shown "
|
|
|
|
|
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-- See Note [show-hide]
|
|
|
|
|
section
|
|
|
|
|
:: Monad m
|
2016-04-11 23:31:30 +03:00
|
|
|
|
=> Text -- ^ Section name (or names)
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-> [Attribute] -- ^ Additional attributes
|
|
|
|
|
-> HtmlT m () -- ^ Content of the section
|
|
|
|
|
-> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
section t attrs = div_ (class_ (t <> " section ") : attrs)
|
|
|
|
|
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-- See Note [show-hide]
|
|
|
|
|
sectionSpan
|
|
|
|
|
:: Monad m
|
2016-04-11 23:31:30 +03:00
|
|
|
|
=> Text -- ^ Section name (or names)
|
2016-03-27 02:34:07 +03:00
|
|
|
|
-> [Attribute] -- ^ Additional attributes
|
|
|
|
|
-> HtmlT m () -- ^ Content of the section
|
|
|
|
|
-> HtmlT m ()
|
2016-03-15 15:35:35 +03:00
|
|
|
|
sectionSpan t attrs = span_ (class_ (t <> " section ") : attrs)
|
|
|
|
|
|
|
|
|
|
newGroupValue :: Text
|
|
|
|
|
newGroupValue = "-new-group-"
|