This commit is contained in:
Aaron Yarborough 2024-03-12 19:51:15 +00:00
parent b99c87201c
commit 81ececda1a
15 changed files with 181 additions and 17 deletions

View file

@ -0,0 +1,8 @@
---
title: JavaScript/HTML5 tile editor
desc: A rough tile editor for a top-down, 2D, tile-based browser game project.
---
I built this 2D tile editor using an HTML 5 canvas and vanilla JS in a few days, as I wanted to create a bespoke editor for my tile game rather than relying on the open-source variants. The reason for this was that I was using my own file format for level files.
[![YouTube thumbnail](https://i3.ytimg.com/vi/7w3fTHYEGbE/maxresdefault.jpg)](https://www.youtube.com/watch?v=7w3fTHYEGbE)
[Watch YouTube video](https://www.youtube.com/watch?v=7w3fTHYEGbE)

View file

@ -22,6 +22,11 @@ languages:
- name: Arabic (Levantine)
proficiency: Elementary
experience:
- position: Lead Consultant
employer: Hippo
start: Feb. 2024
end: Present
desc: "-"
- position: Software Development Tutor
employer: Yarbz Digital Ltd
start: Sep. 2023

1
package-lock.json generated
View file

@ -18,6 +18,7 @@
"@babel/preset-react": "^7.23.3",
"eslint": "^8",
"eslint-config-next": "14.1.1",
"front-matter": "^4.0.2",
"frontmatter-markdown-loader": "^3.7.0",
"husky": "^9.0.11",
"js-yaml": "^4.1.0",

View file

@ -22,6 +22,7 @@
"@babel/preset-react": "^7.23.3",
"eslint": "^8",
"eslint-config-next": "14.1.1",
"front-matter": "^4.0.2",
"frontmatter-markdown-loader": "^3.7.0",
"husky": "^9.0.11",
"js-yaml": "^4.1.0",

View file

@ -15,6 +15,16 @@ collections:
- {label: Image, name: image, widget: image, required: false}
- {label: "You Will Need", name: "you-will-need", widget: "markdown" }
- {label: "Recipe", name: "body", widget: "markdown" }
- name: fun
label: Fun
folder: content/fun
create: true
fields:
- {label: Title, name: title, widget: string}
- {label: Description, name: desc, widget: text}
- {label: Body, name: body, widget: markdown }
- name: "pages"
label: "Pages"
files:

View file

@ -46,9 +46,9 @@ function CVWorkExperience ({ position, employer, start, end, children }) {
<br />
<small>{employer}</small>
</h3>
<div>
<time>{start}</time>-<time>{end}</time>
</div>
<small>
<time>{start}</time> - <time>{end}</time>
</small>
</div>
<div dangerouslySetInnerHTML={{ __html: children }} />
</div>

View file

@ -18,16 +18,6 @@
justify-content: space-between;
}
.cv ul {
margin-left: 0;
list-style: inside;
list-style-type: circle;
}
.cv ul li:not(:last-child) {
margin-bottom: 10px;
}
@media (max-width: 768px) {
.cv {
flex-direction: column;

View file

@ -1,3 +1,7 @@
.footer nav:first-child a {
text-transform: lowercase;
}
}
.footer nav li {
margin-bottom: 0;
}

View file

@ -10,6 +10,7 @@ function Header () {
<Link href='/'>Home</Link>
{/* <Link href='/writing'>Writing</Link> */}
<Link href='/cv'>CV</Link>
<Link href='/fun'>Fun</Link>
</nav>
</header>
)

23
src/lib/content.js Normal file
View file

@ -0,0 +1,23 @@
import fs from 'fs'
import fm from 'front-matter'
import showdown from 'showdown'
import { toSlug } from './helpers'
export function getMarkdownEntry (path) {
const fileContents = fs.readFileSync(path, {
encoding: 'utf-8'
})
const { attributes, body } = fm(fileContents)
const converter = new showdown.Converter()
const html = converter.makeHtml(body)
const slug = toSlug(path.substring(path.lastIndexOf('/')))
return {
attributes,
html,
slug
}
}

3
src/lib/helpers.js Normal file
View file

@ -0,0 +1,3 @@
export function toSlug (input) {
return input.substring(0, input.indexOf('.')).trim()
}

45
src/pages/fun/[slug].js Normal file
View file

@ -0,0 +1,45 @@
import { toSlug } from '@/lib/helpers'
import React from 'react'
import fs from 'fs'
import DefaultLayout from '@/layouts/DefaultLayout/DefaultLayout'
import { getMarkdownEntry } from '@/lib/content'
function FunSingle ({ attributes, html, slug }) {
return (
<DefaultLayout>
<section>
<div />
<h1>{attributes.title}</h1>
<p>{attributes.desc}</p>
</section>
<section>
<div dangerouslySetInnerHTML={{ __html: html }} />
</section>
</DefaultLayout>
)
}
export function getStaticPaths () {
const fun = fs.readdirSync('./content/fun', { withFileTypes: true })
const paths = fun.map((dirent) => ({
params: {
slug: toSlug(dirent.name)
}
}))
console.log(paths)
return {
fallback: false,
paths
}
}
export function getStaticProps ({ params }) {
const path = `./content/fun/${params.slug}.md`
const entry = getMarkdownEntry(path)
return { props: { ...entry } }
}
export default FunSingle

41
src/pages/fun/index.js Normal file
View file

@ -0,0 +1,41 @@
import DefaultLayout from '@/layouts/DefaultLayout/DefaultLayout'
import React from 'react'
import fs from 'fs'
import Grid from '@/components/Grid/Grid'
import Link from 'next/link'
import { getMarkdownEntry } from '@/lib/content'
function Fun ({ entries }) {
return (
<DefaultLayout>
<section>
<h1>Fun</h1>
<p>Hobby projects, helpful scripts, and other fun bits and bobs!</p>
</section>
<section>
<Grid>
{entries.map((e) => (
<div key={e.attributes.title}>
<h2><Link href={'/fun/' + e.slug}>{e.attributes.title}</Link></h2>
<p>{e.attributes.desc}</p>
<Link href={'/fun/' + e.slug}>Read more</Link>
</div>
))}
</Grid>
</section>
</DefaultLayout>
)
}
export function getStaticProps () {
const fun = fs.readdirSync('./content/fun', { withFileTypes: true })
const entries = fun.map((dirent) =>
getMarkdownEntry(`${dirent.path}/${dirent.name}`)
)
return { props: { entries } }
}
export default Fun

View file

@ -21,12 +21,34 @@ export default function Home () {
This is my little corner of the web! I&apos;ve always had a habit of
&apos;lurking&apos; online; I barely interact with the content I
consume, and you&apos;ll rarely if ever catch me posting or commenting
on something. That said, this little site endeavours to pull me by my
ankles out of the weeds in the great digital park we find ourselves,
and encourage me to share a bit more about myself online.
on something. That said, this little site endeavours to encourage me
to share a bit more about myself online.
</p>
</section>
<section>
<h2>Tech I like</h2>
<ul>
<li>
<strong>Sites:</strong> At the moment, I mainly use node with TS (or
JS for small projects) and Next.js to build sites and apps.
</li>
<li>
<strong>Scripts:</strong> My go-to for scripting is either Python or
JS, mainly because I&apos;m comfortable with these languages, and their
library ecosystem usually has everything I need to do what I need to
do.
</li>
<li>
<strong>APIs:</strong> If I need something more robust for API or
back-end architecture than node, I usually go for dotnet core/C#
using ASP.NET. A strongly-typed language with an opinionated web
framework like ASP.NET helps to keep everything neat and tidy.
</li>
<li><strong>Cloud:</strong> If I can get away with it, host on a droplet on Digitalocean. If not, my go-to is usually GCP because it&apos;s less soul-destroying than Azure and AWS (and I&apos;ve used it the most)</li>
</ul>
</section>
<section>
<h2>Where to find me</h2>

View file

@ -6,3 +6,13 @@ html, body {
h1, h2, h3 {
font-weight: 600;
}
ul {
margin-left: 0;
list-style: inside;
list-style-type: circle;
}
ul li:not(:last-child) {
margin-bottom: 10px;
}