Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Hero Image #51

Open
wants to merge 12 commits into
base: main
Choose a base branch
from
4 changes: 4 additions & 0 deletions Application/Helper/Controller.hs
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,11 @@ fetchLandingPageWithRecords :: (?modelContext :: ModelContext) => Id LandingPage
fetchLandingPageWithRecords landingPageId = do

landingPage <- fetch landingPageId

paragraphCtas <- fetch landingPage.paragraphCtasRefLandingPages

paragraphHeroImages <- fetch landingPage.paragraphHeroImages
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved

paragraphQuotes <- fetch landingPage.paragraphQuotes

return $ LandingPageWithRecords { .. }
Expand All @@ -27,6 +30,7 @@ getParagraphsCount landingPageId = do
landingPageWithRecords <- fetchLandingPageWithRecords landingPageId

pure $ length landingPageWithRecords.paragraphCtas
+ length landingPageWithRecords.paragraphHeroImages
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved
+ length landingPageWithRecords.paragraphQuotes
+ 1

Expand Down
12 changes: 10 additions & 2 deletions Application/Schema.sql
Original file line number Diff line number Diff line change
Expand Up @@ -5,15 +5,13 @@ BEGIN
END;
$$ language plpgsql;
-- Your database schema. Use the Schema Designer at http://localhost:8001/ to add some tables.

CREATE TABLE users (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
email TEXT NOT NULL,
password_hash TEXT NOT NULL,
locked_at TIMESTAMP WITH TIME ZONE DEFAULT NULL,
failed_login_attempts INT DEFAULT 0 NOT NULL
);

CREATE TABLE landing_pages (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW() NOT NULL,
Expand All @@ -40,6 +38,16 @@ CREATE TABLE paragraph_ctas (
);
CREATE INDEX paragraph_quotes_landing_page_id_index ON paragraph_quotes (landing_page_id);
CREATE INDEX paragraph_ctas_landing_page_id_index ON paragraph_ctas (landing_page_id);
CREATE TABLE paragraph_hero_images (
id UUID DEFAULT uuid_generate_v4() PRIMARY KEY NOT NULL,
title TEXT NOT NULL,
subtitle TEXT,
landing_page_id UUID DEFAULT uuid_generate_v4() NOT NULL,
link TEXT DEFAULT NULL,
image_url TEXT,
weight INT NOT NULL
);
ALTER TABLE paragraph_ctas ADD CONSTRAINT paragraph_ctas_ref_landing_page_id FOREIGN KEY (landing_page_id) REFERENCES landing_pages (id) ON DELETE NO ACTION;
ALTER TABLE paragraph_ctas ADD CONSTRAINT paragraph_ctas_ref_ref_landing_page_id FOREIGN KEY (ref_landing_page_id) REFERENCES landing_pages (id) ON DELETE NO ACTION;
ALTER TABLE paragraph_hero_images ADD CONSTRAINT paragraph_hero_images_ref_landing_page_id FOREIGN KEY (landing_page_id) REFERENCES landing_pages (id) ON DELETE NO ACTION;
ALTER TABLE paragraph_quotes ADD CONSTRAINT paragraph_quotes_ref_landing_page_id FOREIGN KEY (landing_page_id) REFERENCES landing_pages (id) ON DELETE NO ACTION;
1 change: 1 addition & 0 deletions Web/Controller/LandingPages.hs
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@ instance Controller LandingPagesController where

-- Iterate over all paragraphs, and update the weight.
forEach landingPageWithRecords.paragraphCtas updateParagraph
forEach landingPageWithRecords.paragraphHeroImages updateParagraph
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved
forEach landingPageWithRecords.paragraphQuotes updateParagraph

where
Expand Down
4 changes: 4 additions & 0 deletions Web/Controller/ParagraphCtas.hs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ instance Controller ParagraphCtasController where

-- Get all landing pages, so we can select the one we want to link to.
landingPages <- query @LandingPage |> fetch
landingPage <- fetch paragraphCta.landingPageId

render NewView { .. }

Expand All @@ -23,6 +24,7 @@ instance Controller ParagraphCtasController where

-- Get all landing pages, so we can select the one we want to link to.
landingPages <- query @LandingPage |> fetch
landingPage <- fetch paragraphCta.landingPageId

render EditView { .. }

Expand All @@ -33,6 +35,7 @@ instance Controller ParagraphCtasController where
|> ifValid \case
Left paragraphCta -> do
landingPages <- query @LandingPage |> fetch
landingPage <- fetch paragraphCta.landingPageId
render EditView { .. }
Right paragraphCta -> do
paragraphCta <- paragraphCta |> updateRecord
Expand All @@ -46,6 +49,7 @@ instance Controller ParagraphCtasController where
|> ifValid \case
Left paragraphCta -> do
landingPages <- query @LandingPage |> fetch
landingPage <- fetch paragraphCta.landingPageId
render NewView { .. }
Right paragraphCta -> do
paragraphCta <- paragraphCta |> createRecord
Expand Down
72 changes: 72 additions & 0 deletions Web/Controller/ParagraphHeroImages.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
module Web.Controller.ParagraphHeroImages where

import Web.Controller.Prelude
import Web.View.ParagraphHeroImages.New
import Web.View.ParagraphHeroImages.Edit

instance Controller ParagraphHeroImagesController where
action NewParagraphHeroImageAction { .. } = do
weight <- getParagraphsCount landingPageId
let paragraphHeroImage = newRecord
|> set #landingPageId landingPageId
|> set #weight weight

let formStatus = FormStatusNotSubmitted
landingPage <- fetch paragraphHeroImage.landingPageId

render NewView { .. }

action EditParagraphHeroImageAction { paragraphHeroImageId } = do
paragraphHeroImage <- fetch paragraphHeroImageId
-- Get from the session, if the form was submitted successfully.
formStatus <- getAndClearFormStatus
landingPage <- fetch paragraphHeroImage.landingPageId

render EditView { .. }

action UpdateParagraphHeroImageAction { paragraphHeroImageId } = do
createOrUpdateParagraphHeroImageAction (Just paragraphHeroImageId)

action CreateParagraphHeroImageAction = do
createOrUpdateParagraphHeroImageAction Nothing

action DeleteParagraphHeroImageAction { paragraphHeroImageId } = do
paragraphHeroImage <- fetch paragraphHeroImageId
deleteRecord paragraphHeroImage
setSuccessMessage "Hero Image deleted"
redirectTo EditLandingPageAction { landingPageId = paragraphHeroImage.landingPageId }

buildParagraphHeroImage paragraphHeroImage = paragraphHeroImage
|> fill @["landingPageId", "weight", "title", "subtitle", "link"]
|> validateField #title nonEmpty
|> validateField #imageUrl nonEmpty
|> pure

createOrUpdateParagraphHeroImageAction :: (?modelContext :: ModelContext, ?context :: ControllerContext) => Maybe (Id ParagraphHeroImage) -> IO ()
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved
createOrUpdateParagraphHeroImageAction maybeParagraphHeroImageId = do
let uploadImage = uploadToStorageWithOptions $ def
{ preprocess = applyImageMagick "jpg" ["-resize", "1024x1024^", "-gravity", "north", "-extent", "1024x1024", "-quality", "85%", "-strip"] }

formStatus <- getAndClearFormStatus

paragraphHeroImage <- case maybeParagraphHeroImageId of
Just id -> fetch id
Nothing -> pure $ newRecord @ParagraphHeroImage

paragraphHeroImage
|> uploadImage #imageUrl
>>= buildParagraphHeroImage
>>= ifValid \case
Left paragraphHeroImage -> do
setFormStatus FormStatusError
landingPage <- fetch paragraphHeroImage.landingPageId
if isJust maybeParagraphHeroImageId
then render EditView { .. }
else render NewView { .. }
Right paragraphHeroImage -> do
paragraphHeroImage <- case maybeParagraphHeroImageId of
Just _ -> paragraphHeroImage |> updateRecord
Nothing -> paragraphHeroImage |> createRecord
setSuccessMessage "Hero Image saved"
-- We don't setFormStatus, since we redirect to a new page.
redirectTo EditLandingPageAction { landingPageId = paragraphHeroImage.landingPageId }
68 changes: 32 additions & 36 deletions Web/Controller/ParagraphQuotes.hs
Original file line number Diff line number Diff line change
Expand Up @@ -12,54 +12,22 @@ instance Controller ParagraphQuotesController where
|> set #weight weight

let formStatus = FormStatusNotSubmitted
landingPage <- fetch paragraphQuote.landingPageId

render NewView { .. }

action EditParagraphQuoteAction { paragraphQuoteId } = do
paragraphQuote <- fetch paragraphQuoteId
landingPage <- fetch paragraphQuote.landingPageId
-- Get from the session, if the form was submitted successfully.
formStatus <- getAndClearFormStatus
render EditView { .. }

action UpdateParagraphQuoteAction { paragraphQuoteId } = do
let uploadImage = uploadToStorageWithOptions $ def
{ preprocess = applyImageMagick "jpg" ["-resize", "1024x1024^", "-gravity", "north", "-extent", "1024x1024", "-quality", "85%", "-strip"] }

formStatus <- getAndClearFormStatus

paragraphQuote <- fetch paragraphQuoteId
paragraphQuote
|> uploadImage #imageUrl
>>= buildParagraphQuote
>>= ifValid \case
Left paragraphQuote -> do
setFormStatus FormStatusError
render EditView { .. }
Right paragraphQuote -> do
paragraphQuote <- paragraphQuote |> updateRecord
setSuccessMessage "Quote updated"
-- We don't setFormStatus, since we redirect to a new page.
redirectTo EditLandingPageAction { landingPageId = paragraphQuote.landingPageId }
createOrUpdateParagraphQuoteAction (Just paragraphQuoteId)

action CreateParagraphQuoteAction = do
let uploadImage = uploadToStorageWithOptions $ def
{ preprocess = applyImageMagick "jpg" ["-resize", "1024x1024^", "-gravity", "north", "-extent", "1024x1024", "-quality", "85%", "-strip"] }

let paragraphQuote = newRecord @ParagraphQuote
let formStatus = FormStatusNotSubmitted

paragraphQuote
|> uploadImage #imageUrl
>>= buildParagraphQuote
>>= ifValid \case
Left paragraphQuote -> do
setFormStatus FormStatusError
render NewView { .. }
Right paragraphQuote -> do
paragraphQuote <- paragraphQuote |> createRecord
setSuccessMessage "Quote created"
-- We don't setFormStatus, since we redirect to a new page.
redirectTo EditLandingPageAction { landingPageId = paragraphQuote.landingPageId }
createOrUpdateParagraphQuoteAction Nothing

action DeleteParagraphQuoteAction { paragraphQuoteId } = do
paragraphQuote <- fetch paragraphQuoteId
Expand All @@ -74,3 +42,31 @@ buildParagraphQuote paragraphQuote = paragraphQuote
|> validateField #imageUrl nonEmpty
|> return

createOrUpdateParagraphQuoteAction :: (?modelContext :: ModelContext, ?context :: ControllerContext) => Maybe (Id ParagraphQuote) -> IO ()
createOrUpdateParagraphQuoteAction maybeParagraphQuoteId = do
let uploadImage = uploadToStorageWithOptions $ def
{ preprocess = applyImageMagick "jpg" ["-resize", "1024x1024^", "-gravity", "north", "-extent", "1024x1024", "-quality", "85%", "-strip"] }

formStatus <- getAndClearFormStatus

paragraphQuote <- case maybeParagraphQuoteId of
Just id -> fetch id
Nothing -> pure $ newRecord @ParagraphQuote

paragraphQuote
|> uploadImage #imageUrl
>>= buildParagraphQuote
>>= ifValid \case
Left paragraphQuote -> do
setFormStatus FormStatusError
landingPage <- fetch paragraphQuote.landingPageId
if isJust maybeParagraphQuoteId
then render EditView { .. }
else render NewView { .. }
Right paragraphQuote -> do
paragraphQuote <- case maybeParagraphQuoteId of
Just _ -> paragraphQuote |> updateRecord
Nothing -> paragraphQuote |> createRecord
setSuccessMessage "Hero Image saved"
-- We don't setFormStatus, since we redirect to a new page.
redirectTo EditLandingPageAction { landingPageId = paragraphQuote.landingPageId }
1 change: 1 addition & 0 deletions Web/Element/ElementWrap.hs
Original file line number Diff line number Diff line change
Expand Up @@ -332,6 +332,7 @@ wrapTextFontWeight fontweight element =
case fontweight of
FontWeightLight -> "font-light"
FontWeightNormal -> "font-normal"
FontWeightMedium -> "font-medium"
FontWeightBold -> "font-bold"

wrapHeaderTag :: Int -> Html -> Html
Expand Down
55 changes: 55 additions & 0 deletions Web/Element/HeroImage.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
module Web.Element.HeroImage where

import Web.View.Prelude

import Web.Element.ElementWrap
import Web.Element.Types
import Web.Element.Button

render :: RenderHeroImage -> Html
render record@RenderHeroImage {title, maybeSubtitle, maybeButton, imageUrl} =
titleWrapped
++ subTitleWrapped
++ renderedButton
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved
|> wrapVerticalSpacingBig AlignNone
|> renderImageAndContent (pathTo $ RenderImageStyleAction 1536 466 imageUrl signed)
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved
where
-- Sign the image URL to prevent tampering.
signed = signImageUrl imageUrl 1536 466

titleWrapped = cs title
|> wrapHeaderTag 1
|> wrapTextFontWeight FontWeightBold
|> wrapTextResponsiveFontSize TextSizeSm
andreytroeglazov marked this conversation as resolved.
Show resolved Hide resolved

subTitleWrapped = maybeSubtitle
|> fmap (\subtitle -> cs subtitle
|> wrapTextFontWeight FontWeightMedium
|> wrapTextResponsiveFontSize TextSizeXl
)
|> fromMaybe ""

renderedButton = maybeButton |> fmap Web.Element.Button.render |> fromMaybe ""


renderImageAndContent :: Text -> Html -> Html
renderImageAndContent imageUrl items =
[hsx|
<div class="grid grid-rows-1">
<figure class="row-start-1 col-start-1 child-object-cover">
{image}
</figure>

<div class="row-start-1 col-start-1 z-10 flex flex-col justify-center">
<div class="container-wide w-full">
<div class="max-w-prose my-6 md:my-8 p-6 md:p-8 lg:p-10 bg-white/90 flex flex-col gap-3 md:gap-4 lg:gap-5 rounded-lg">
{items}
</div>
</div>
</div>
</div>
|]
where
image = [hsx|<img src={imageUrl} class="w-full h-full" />|]


9 changes: 8 additions & 1 deletion Web/Element/Types.hs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ data Align = AlignNone | AlignStart | AlignCenter | AlignEnd | AlignBaseline der

data Justify = JustifyStart | JustifyCenter | JustifyEnd | JustifyBetween deriving (Eq, Show)

data FontWeight = FontWeightLight | FontWeightNormal | FontWeightBold deriving (Eq, Show)
data FontWeight = FontWeightLight | FontWeightNormal | FontWeightMedium | FontWeightBold deriving (Eq, Show)

data Color
= ColorTransparent
Expand Down Expand Up @@ -60,6 +60,13 @@ data RenderQuote = RenderQuote
, imageUrl :: Text
}

data RenderHeroImage = RenderHeroImage
{ title :: Text
, maybeSubtitle :: Maybe Text
, imageUrl :: Text
, maybeButton :: Maybe RenderButton
}


-- @todo: Move to Utility.hs?
getBackgroundColor :: Color -> Text
Expand Down
2 changes: 2 additions & 0 deletions Web/FrontController.hs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import Web.Controller.ImageStyle
import Web.Controller.LandingPages
import Web.Controller.ParagraphCtas
import Web.Controller.ParagraphQuotes
import Web.Controller.ParagraphHeroImages

import IHP.LoginSupport.Middleware
import Web.Controller.Sessions
Expand All @@ -26,6 +27,7 @@ instance FrontController WebApplication where
, parseRoute @LandingPagesController
, parseRoute @ParagraphCtasController
, parseRoute @ParagraphQuotesController
, parseRoute @ParagraphHeroImagesController
, parseRoute @SessionsController
]

Expand Down
3 changes: 1 addition & 2 deletions Web/Routes.hs
Original file line number Diff line number Diff line change
Expand Up @@ -7,13 +7,12 @@ import Web.Types

instance AutoRoute ParagraphQuotesController


instance AutoRoute ParagraphCtasController

instance AutoRoute ParagraphHeroImagesController

instance AutoRoute LandingPagesController


instance AutoRoute ImageStyleController

instance AutoRoute SessionsController
Expand Down
9 changes: 9 additions & 0 deletions Web/Types.hs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ data LandingPageWithRecords = LandingPageWithRecords
{ landingPage :: !LandingPage
, paragraphCtas :: ![ParagraphCta]
, paragraphQuotes :: ![ParagraphQuote]
, paragraphHeroImages :: ![ParagraphHeroImage]
} deriving (Show)

{-| With the `FormStatus` we can show a message to the user after submitting a form,
Expand Down Expand Up @@ -64,6 +65,14 @@ data ParagraphQuotesController
| DeleteParagraphQuoteAction { paragraphQuoteId :: !(Id ParagraphQuote) }
deriving (Eq, Show, Data)

data ParagraphHeroImagesController
= NewParagraphHeroImageAction { landingPageId :: !(Id LandingPage) }
| CreateParagraphHeroImageAction
| EditParagraphHeroImageAction { paragraphHeroImageId :: !(Id ParagraphHeroImage) }
| UpdateParagraphHeroImageAction { paragraphHeroImageId :: !(Id ParagraphHeroImage) }
| DeleteParagraphHeroImageAction { paragraphHeroImageId :: !(Id ParagraphHeroImage) }
deriving (Eq, Show, Data)



data ImageStyleController
Expand Down
Loading
Loading