1
1
mirror of https://github.com/aelve/guide.git synced 2024-12-25 05:43:32 +03:00

Move JQuerySelector into JS.hs

This commit is contained in:
Artyom 2016-03-15 13:10:47 +03:00
parent 46ace5f57a
commit 993e33d154
3 changed files with 40 additions and 41 deletions

View File

@ -24,6 +24,7 @@ import qualified Data.Text.Buildable as Format
import NeatInterpolation import NeatInterpolation
-- Local -- Local
import Types
import Utils import Utils
@ -66,6 +67,8 @@ instance ToJS Integer where
toJS = JS . tshow toJS = JS . tshow
instance ToJS Int where instance ToJS Int where
toJS = JS . tshow toJS = JS . tshow
instance ToJS Uid where
toJS = toJS . uidToText
-- | A helper class for calling Javascript functions. -- | A helper class for calling Javascript functions.
class JSParams a where class JSParams a where
@ -424,3 +427,21 @@ escapeJSString s =
B.singleton c B.singleton c
where where
h = showHex (fromEnum c) "" h = showHex (fromEnum c) ""
newtype JQuerySelector = JQuerySelector Text
deriving (ToJS, Format.Buildable)
selectId :: Text -> JQuerySelector
selectId x = JQuerySelector $ format "#{}" [x]
selectUid :: Uid -> JQuerySelector
selectUid x = JQuerySelector $ format "#{}" [x]
selectClass :: Text -> JQuerySelector
selectClass x = JQuerySelector $ format ".{}" [x]
selectParent :: JQuerySelector -> JQuerySelector
selectParent x = JQuerySelector $ format ":has(> {})" [x]
selectChildren :: JQuerySelector -> JQuerySelector -> JQuerySelector
selectChildren a b = JQuerySelector $ format "{} > {}" (a, b)

View File

@ -27,7 +27,6 @@ import qualified Data.Text as T
import qualified Data.Text.Encoding as T import qualified Data.Text.Encoding as T
import qualified Data.Text.IO as T import qualified Data.Text.IO as T
import NeatInterpolation import NeatInterpolation
import qualified Data.Text.Buildable as Format
-- Randomness -- Randomness
import System.Random import System.Random
-- Web -- Web
@ -48,7 +47,7 @@ import Data.Acid as Acid
-- Local -- Local
import Types import Types
import JS (JS(..), ToJS(..), allJSFunctions) import JS (JS(..), allJSFunctions, JQuerySelector)
import qualified JS import qualified JS
import Utils import Utils
import Markdown import Markdown
@ -304,9 +303,9 @@ main = do
Spock.get "donate" $ do Spock.get "donate" $ do
lucid $ renderDonate lucid $ renderDonate
-- The add/set methods return rendered parts of the structure (added -- The add/set methods return rendered parts of the structure (added
-- categories, changed items, etc) so that the Javascript part could take -- categories, changed items, etc) so that the Javascript part could
-- them and inject into the page. We don't want to duplicate rendering on -- take them and inject into the page. We don't want to duplicate
-- server side and on client side. -- rendering on server side and on client side.
renderMethods renderMethods
setMethods setMethods
addMethods addMethods
@ -381,14 +380,14 @@ renderRoot globalState mbSearchQuery = doctypehtml_ $ do
you won't be able to edit anything. you won't be able to edit anything.
|] |]
renderHelp renderHelp
onPageLoad $ JS.showOrHideHelp (selectId "help", helpVersion) onPageLoad $ JS.showOrHideHelp (JS.selectId "help", helpVersion)
form_ $ do form_ $ do
input_ [type_ "text", name_ "q", id_ "search", placeholder_ "search", input_ [type_ "text", name_ "q", id_ "search", placeholder_ "search",
value_ (fromMaybe "" mbSearchQuery)] value_ (fromMaybe "" mbSearchQuery)]
textInput [ textInput [
placeholder_ "add a category", placeholder_ "add a category",
autocomplete_ "off", autocomplete_ "off",
onEnter $ JS.addCategory (selectId "categories", inputValue) <> onEnter $ JS.addCategory (JS.selectId "categories", inputValue) <>
clearInput ] clearInput ]
-- TODO: sort categories by popularity, somehow? or provide a list of -- TODO: sort categories by popularity, somehow? or provide a list of
-- “commonly used categories” or even a nested catalog -- “commonly used categories” or even a nested catalog
@ -496,11 +495,11 @@ renderHelp = do
-- 'JS.hideHelp' -- 'JS.hideHelp'
section "collapsed" [shown] $ do section "collapsed" [shown] $ do
textButton "show help" $ textButton "show help" $
JS.showHelp (selectId "help", helpVersion) JS.showHelp (JS.selectId "help", helpVersion)
section "expanded" [noScriptShown] $ do section "expanded" [noScriptShown] $ do
textButton "hide help" $ textButton "hide help" $
JS.hideHelp (selectId "help", helpVersion) JS.hideHelp (JS.selectId "help", helpVersion)
-- Don't forget to change 'helpVersion' when the text changes -- Don't forget to change 'helpVersion' when the text changes
-- substantially and you think the users should reread it -- substantially and you think the users should reread it
help <- liftIO $ T.readFile "static/help.md" help <- liftIO $ T.readFile "static/help.md"
@ -518,8 +517,6 @@ helpVersion = 2
-- TODO: automatic merge should be possible too (e.g. if the changes are in -- TODO: automatic merge should be possible too (e.g. if the changes are in
-- different paragraphs) -- different paragraphs)
-- TODO: [very-easy] rename selectChild to selectChildren
renderCategoryList :: [Category] -> HtmlT IO () renderCategoryList :: [Category] -> HtmlT IO ()
renderCategoryList cats = renderCategoryList cats =
div_ [id_ "categories"] $ div_ [id_ "categories"] $
@ -528,7 +525,7 @@ renderCategoryList cats =
renderCategoryTitle :: Category -> HtmlT IO () renderCategoryTitle :: Category -> HtmlT IO ()
renderCategoryTitle category = do renderCategoryTitle category = do
let thisId = "category-title-" <> uidToText (category^.uid) let thisId = "category-title-" <> uidToText (category^.uid)
this = selectId thisId this = JS.selectId thisId
-- TODO: once pagination or something is implemented, we'll have to see -- TODO: once pagination or something is implemented, we'll have to see
-- whether an anchor has been used in the query string and load the -- whether an anchor has been used in the query string and load the
-- necessary category if so -- necessary category if so
@ -554,7 +551,7 @@ renderCategoryTitle category = do
renderCategoryNotes :: Category -> HtmlT IO () renderCategoryNotes :: Category -> HtmlT IO ()
renderCategoryNotes category = do renderCategoryNotes category = do
let thisId = "category-notes-" <> uidToText (category^.uid) let thisId = "category-notes-" <> uidToText (category^.uid)
this = selectId thisId this = JS.selectId thisId
div_ [id_ thisId] $ do div_ [id_ thisId] $ do
section "normal" [shown, noScriptShown] $ do section "normal" [shown, noScriptShown] $ do
@ -614,7 +611,7 @@ renderItemInfo :: Category -> Item -> HtmlT IO ()
renderItemInfo cat item = do renderItemInfo cat item = do
let bg = hueToDarkColor $ getItemHue cat item let bg = hueToDarkColor $ getItemHue cat item
let thisId = "item-info-" <> uidToText (item^.uid) let thisId = "item-info-" <> uidToText (item^.uid)
this = selectId thisId this = JS.selectId thisId
div_ [id_ thisId, class_ "item-info", div_ [id_ thisId, class_ "item-info",
style_ ("background-color:" <> bg)] $ do style_ ("background-color:" <> bg)] $ do
@ -652,7 +649,7 @@ renderItemInfo cat item = do
emptySpan "2em" emptySpan "2em"
toHtml (fromMaybe "other" (item^.group_)) toHtml (fromMaybe "other" (item^.group_))
span_ [class_ "controls"] $ do span_ [class_ "controls"] $ do
let itemNode = selectId ("item-" <> uidToText (item^.uid)) let itemNode = JS.selectId ("item-" <> uidToText (item^.uid))
imgButton "move item up" "/arrow-thick-top.svg" [] $ imgButton "move item up" "/arrow-thick-top.svg" [] $
-- TODO: [easy] the item should blink or somehow else show where it -- TODO: [easy] the item should blink or somehow else show where it
-- has been moved -- has been moved
@ -672,8 +669,8 @@ renderItemInfo cat item = do
let selectedIf p x = if p then with x [selected_ "selected"] else x let selectedIf p x = if p then with x [selected_ "selected"] else x
-- otherNodes are all nodes that have to be recolored when this node is -- otherNodes are all nodes that have to be recolored when this node is
-- recolored -- recolored
let otherNodes = selectChild (selectParent this) let otherNodes = JS.selectChildren (JS.selectParent this)
(selectClass "item-body") (JS.selectClass "item-body")
let formSubmitHandler formNode = let formSubmitHandler formNode =
JS.submitItemInfo (this, otherNodes, item^.uid, formNode) JS.submitItemInfo (this, otherNodes, item^.uid, formNode)
form_ [onFormSubmit formSubmitHandler] $ do form_ [onFormSubmit formSubmitHandler] $ do
@ -756,7 +753,7 @@ renderItemDescription category item = do
-- 'otherNodes' selector in 'renderItemInfo'. Specifically, we depend on -- 'otherNodes' selector in 'renderItemInfo'. Specifically, we depend on
-- having a div with a class “item-body” here. -- having a div with a class “item-body” here.
let thisId = "item-description-" <> uidToText (item^.uid) let thisId = "item-description-" <> uidToText (item^.uid)
this = selectId thisId this = JS.selectId thisId
div_ [id_ thisId, class_ "item-description item-body", div_ [id_ thisId, class_ "item-description item-body",
style_ ("background-color:" <> bg)] $ do style_ ("background-color:" <> bg)] $ do
@ -820,7 +817,7 @@ renderTrait :: Uid -> Trait -> HtmlT IO ()
-- TODO: probably use renderMarkdownBlock here as well -- TODO: probably use renderMarkdownBlock here as well
renderTrait itemId trait = do renderTrait itemId trait = do
let thisId = "trait-" <> uidToText (trait^.uid) let thisId = "trait-" <> uidToText (trait^.uid)
this = selectId thisId this = JS.selectId thisId
li_ [id_ thisId] $ do li_ [id_ thisId] $ do
sectionSpan "normal" [shown, noScriptShown] $ do sectionSpan "normal" [shown, noScriptShown] $ do
@ -880,7 +877,7 @@ renderItemNotes category item = do
-- 'otherNodes' selector in 'renderItemInfo'. Specifically, we depend on -- 'otherNodes' selector in 'renderItemInfo'. Specifically, we depend on
-- having a div with a class “item-body” here. -- having a div with a class “item-body” here.
let thisId = "item-notes-" <> uidToText (item^.uid) let thisId = "item-notes-" <> uidToText (item^.uid)
this = selectId thisId this = JS.selectId thisId
div_ [id_ thisId, class_ "item-notes item-body", div_ [id_ thisId, class_ "item-notes item-body",
style_ ("background-color:" <> bg)] $ do style_ ("background-color:" <> bg)] $ do
-- TODO: this duplicates code from renderCategoryNotes, try to reduce -- TODO: this duplicates code from renderCategoryNotes, try to reduce
@ -1017,31 +1014,13 @@ smallMarkdownEditor attributes s submit mbCancel = do
uid_ :: Uid -> Attribute uid_ :: Uid -> Attribute
uid_ = id_ . uidToText uid_ = id_ . uidToText
newtype JQuerySelector = JQuerySelector Text
deriving (ToJS, Format.Buildable)
selectId :: Text -> JQuerySelector
selectId x = JQuerySelector $ format "#{}" [x]
selectUid :: Uid -> JQuerySelector
selectUid x = JQuerySelector $ format "#{}" [x]
selectClass :: Text -> JQuerySelector
selectClass x = JQuerySelector $ format ".{}" [x]
selectParent :: JQuerySelector -> JQuerySelector
selectParent x = JQuerySelector $ format ":has(> {})" [x]
selectChild :: JQuerySelector -> JQuerySelector -> JQuerySelector
selectChild a b = JQuerySelector $ format "{} > {}" (a, b)
thisNode :: HtmlT IO JQuerySelector thisNode :: HtmlT IO JQuerySelector
thisNode = do thisNode = do
uid' <- randomUid uid' <- randomUid
-- If the class name ever changes, fix 'JS.moveNodeUp' and -- If the class name ever changes, fix 'JS.moveNodeUp' and
-- 'JS.moveNodeDown'. -- 'JS.moveNodeDown'.
span_ [uid_ uid', class_ "dummy"] mempty span_ [uid_ uid', class_ "dummy"] mempty
return (selectParent (selectUid uid')) return (JS.selectParent (JS.selectUid uid'))
-- TODO: add an “ecosystem” field with related packages/etc (just a simple -- TODO: add an “ecosystem” field with related packages/etc (just a simple
-- Markdown-edited field under pros/cons) -- Markdown-edited field under pros/cons)

View File

@ -101,13 +101,12 @@ import Data.Acid as Acid
-- Local -- Local
import Utils import Utils
import JS (ToJS)
-- | Unique id, used for many things categories, items, and anchor ids. -- | Unique id, used for many things categories, items, and anchor ids.
-- Note that in HTML 5 using numeric ids for divs, spans, etc is okay. -- Note that in HTML 5 using numeric ids for divs, spans, etc is okay.
newtype Uid = Uid {uidToText :: Text} newtype Uid = Uid {uidToText :: Text}
deriving (Eq, PathPiece, ToJS, Format.Buildable, Data) deriving (Eq, PathPiece, Format.Buildable, Data)
deriveSafeCopy 0 'base ''Uid deriveSafeCopy 0 'base ''Uid