Lambda functions are really good at returning a string of text. In this article, we will review how to create a cool personal website with nothing but functions. We won't have to worry about configuring a web server, and our content will be server-side rendered, but serverlessly.
To get started with the completed app click the button below to deploy it in 15 seconds:
Our entire page will be rendered from a single lambda function. Take a look at src/http/get-index/mod.ts
import Main from './views/main.js'
export async function handler (/*req: object*/) {
return {
statusCode: 200,
headers: {
'content-type': 'text/html; charset=utf8',
'cache-control': 'no-cache, no-store, must-revalidate, max-age=0, s-maxage=0'
},
body: Main({
/**
* Basic bio
*/
fullname: 'Your Name', // ← Start by adding your name!
title: 'My personal site!',
occupation: 'Artist & Photographer',
location: 'West Glacier, MT',
bio: 'Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Quis ipsum suspendisse ultrices gravida. Risus commodo viverra maecenas accumsan lacus vel facilisis.',
/**
* Contact / social
* - Comment out any item below to remove it from your page
*/
email: 'your@email.com',
twitter: 'yourTwitter',
linkedin: 'your-linkedin-name',
instagram: 'yourInsta',
facebook: 'your-facebook-name',
/**
* Layout
*/
photographer: 'Ivana Cajina',
service: 'Unsplash',
credit: 'https://unsplash.com/@von_co',
image: '_static/background.jpg'
})
}
}
This single Lambda function takes in the information we will need to fill in the site. Notice that we can use ESM to import the Main
function, which will return an HTML string to the browser.
Let's take a look at the Main
function that we import.
import Styles from './styles.js'
import Symbols from './symbols.js'
import Splash from './splash.js'
import Content from './content.js'
export default function Main(props = {}) {
let title = props.title || 'Personal Website'
return `
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, minimum-scale=1, initial-scale=1">
<title>${title}</title>
${Styles(props)}
<!-- Replace this with your own custom font link and edit Styles font-family -->
<link href="https://fonts.googleapis.com/css?family=Roboto:300,400" rel="stylesheet">
<!-- End custom font -->
<link href="" rel="icon" type="image/x-icon">
</head>
<body
class="
min-width-20
display-flex-large
height-100vh
overflow-hidden-large
"
>
${Symbols}
${Splash(props)}
${Content(props)}
</body>
</html>
`
}
From here the site is broken up into different modules for each component. Each component is a function that accepts props and returns an HTML string. The modules we import at the top are going to be all the styles, svg symbols, a component for the background image and accreditation, and finally the content.
The pattern found here can be traced all the way down through content.js
, which contains more submodules like heading-large.js
, which is the component for showing a name. Let's take a look at content.js
and heading-large.js
.
import LargeHeading from './heading-large.js'
import MediumHeading from './heading-medium.js'
import LocationLink from './link-location.js'
import MailLink from './link-mail.js'
import SocialMedia from './social-media.js'
import Icon from './icon.js'
export default function Content(props = {}) {
let fullname = props.fullname || ''
let occupation = props.occupation || ''
let location = props.location || ''
let bio = props.bio || ''
let email = props.email || ''
let twitter = props.twitter || ''
let linkedin = props.linkedin || ''
let instagram = props.instagram || ''
let facebook = props.facebook || ''
return `
<section
class="
display-flex
flex-direction-column
height-content
height-auto-large
overflow-auto-large
"
>
<div
class="
display-flex
align-items-center-large
justify-content-center-large
flex-grow-1
flex-grow-2-large
max-width-35
margin-right-auto
margin-left-auto
padding-48
padding-5-large
"
>
<div
class="
margin-right-auto
margin-left-auto
"
>
${LargeHeading({
children: fullname
})}
${MediumHeading({
children: occupation
})}
${LocationLink({
location
})}
<p
class="
margin-bottom-42
font-size-16
color-383D3B
"
>
${bio}
</p>
<div
class="
display-flex
flex-wrap-wrap
align-items-center
justify-content-space-between
margin-bottom-16
"
>
${MailLink({
email
})}
${SocialMedia({
twitter,
linkedin,
instagram,
facebook
})}
</div>
</div>
</div>
<div
class="
display-flex
align-items-center
justify-content-space-between
padding-top-16
padding-right-32
padding-left-32
padding-right-48-large
padding-bottom-16
padding-left-48-large
color-5A5C5B
background-color-F2F0F3
"
>
<span
class="
display-flex
align-item-center
"
>
<span
class="
margin-right-8
color-979797
"
>
Built with
</span>
<a
class="
fill-979797
fill-hover-FD6D6D
transition-fill
"
href="https://begin.com"
target="_blank"
rel="noopener"
>
${Icon({
class: 'fill-inherit',
href: 'begin',
style: 'width:4rem;height:1.2725rem;'
})}
</a>
</span>
<a
class="
display-block
padding-top-8
padding-right-16
padding-bottom-8
padding-left-16
font-size-12
font-weight-300
text-decoration-none
color-FFFFFF
border-radius-pill
background-color-979797
background-color-hover-058AEA
transition-background-color
text-transform-uppercase
"
href="https://begin.com"
rel="noopener"
target="_blank"
>
Build yours
</a>
</div>
</section>
`
}
The Content()
function calls LargeHeading()
and passes in the name to be rendered. Let's look at heading-large.js
.
export default function LargeHeading(props = {}) {
let children = props.children || ''
return `
<h1
class="
margin-bottom-24
font-size-42
font-weight-300
color-5A5C5B
"
>
${children}
</h1>
`
}
We can follow this same pattern for the rest of the submodules in content.js
where each function is returning a string template literal of HTML.
We can now get clean HTML to the front end with dynamic information constructed from a single Lambda function.
Top comments (0)