diff --git a/.gitignore b/.gitignore
index 9cfc02d..00f8379 100644
--- a/.gitignore
+++ b/.gitignore
@@ -63,4 +63,5 @@ typings/
/nbproject/private/
.vscode
.serverless
-.cache
\ No newline at end of file
+.cache
+yarn.lock
\ No newline at end of file
diff --git a/.travis.yml b/.travis.yml
deleted file mode 100644
index 9afebea..0000000
--- a/.travis.yml
+++ /dev/null
@@ -1,32 +0,0 @@
-language: node_js
-
-node_js:
- - "10.16.3"
-
-cache:
- yarn: true
- directories:
- - node_modules
-
-matrix:
- include:
- - env: PROJECT=src/private/
- include:
- - env: PROJECT=src/public/
-
-install:
- - cd $PROJECT
- - yarn global add serverless
- - yarn install
-
-script:
- - yarn test
- - yarn build
-
-deploy:
- provider: script
- script:
- - yarn deploy
- skip_cleanup: true
- on:
- branch: master
\ No newline at end of file
diff --git a/main.yml b/main.yml
new file mode 100644
index 0000000..f04d624
--- /dev/null
+++ b/main.yml
@@ -0,0 +1,25 @@
+name: CI
+on:
+ push:
+ branches: [ master ]
+ pull_request:
+ branches: [ master ]
+ workflow_dispatch:
+
+jobs:
+ build-and-deploy:
+ runs-on: ubuntu-latest
+ steps:
+ - name: Checkout 🛎️
+ uses: actions/checkout@v2.3.1
+
+ - name: Install and Build
+ run: |
+ npm install
+ npm run compile
+
+ - name: Deploy
+ uses: JamesIves/github-pages-deploy-action@4.1.0
+ with:
+ branch: gh-pages
+ folder: dist
\ No newline at end of file
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..f197a35
--- /dev/null
+++ b/package.json
@@ -0,0 +1,16 @@
+{
+ "name": "domsplace",
+ "version": "7.0.0",
+ "main": "index.js",
+ "license": "MIT",
+ "scripts": {
+ "start": "ts-node ./src/serve",
+ "compile": "ts-node ./src/compile"
+ },
+ "dependencies": {
+ "@types/express": "^4.17.11",
+ "express": "^4.17.1",
+ "ts-node": "^9.1.1",
+ "typescript": "^4.2.3"
+ }
+}
diff --git a/src/compile.ts b/src/compile.ts
new file mode 100644
index 0000000..2515c5d
--- /dev/null
+++ b/src/compile.ts
@@ -0,0 +1,3 @@
+import { pageCompileAll } from "./compiler";
+
+pageCompileAll();
\ No newline at end of file
diff --git a/src/compiler.ts b/src/compiler.ts
new file mode 100644
index 0000000..ae97eff
--- /dev/null
+++ b/src/compiler.ts
@@ -0,0 +1,75 @@
+// Copyright (c) 2021 Dominic Masters
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+
+import * as fs from 'fs';
+import * as path from 'path';
+
+// Path Definitions
+export const PATH_HTML = path.join(__dirname, 'html');
+export const PATH_CSS = path.join(__dirname, 'css');
+export const PATH_OUT = path.join(__dirname, '..', 'dist');
+
+// Load Methods
+const loadFile = (dir:string, ext:string) => {
+ const bits = dir.split('/').join('\\').split('\\');
+ const fileName = bits.pop();
+ const fullDir = path.join(__dirname, ...bits, `${fileName}.${ext}`);
+
+ if(!fs.existsSync(fullDir)) return '';
+ return fs.readFileSync(fullDir, 'utf-8');
+}
+
+const loadHtml = (fn:string) => loadFile(fn, 'html');
+const loadCss = (fn:string) => loadFile(fn, 'css');
+
+const pageCompile = (page:string) => {
+ // Load Layout
+ const htmlLayout = loadHtml('partials/layout/layout');
+ const cssLayout = loadCss('partials/layout/layout');
+
+ // Load Page
+ const htmlPage = loadHtml(`pages/${page}/${page}`);
+ const cssPage = loadCss(`pages/${page}/${page}`);
+
+ // Inject styles
+ let compiled = htmlLayout.replace('{{styles}}', [
+ cssLayout,
+ cssPage
+ ].join('\n'));
+
+ // Inject content
+ compiled = compiled.replace('{{main}}', htmlPage);
+
+ // Inject variables
+ Object.entries({
+ 'year': new Date().getFullYear()
+ }).forEach(entry => {
+ const [ key, value ] = entry;
+ const variable = `{{${key}}}`;
+ while(compiled.includes(variable)) {
+ compiled = compiled.replace(variable, `${value}`);
+ }
+ });
+
+ return compiled;
+}
+
+const pageGenerate = (out:string, page:string) => {
+ const compiled = pageCompile(page);
+ console.log(page, '->', out);
+ if(!fs.existsSync(PATH_OUT)) fs.mkdirSync(PATH_OUT);
+ fs.writeFileSync(path.join(PATH_OUT, `${out}.html`), compiled);
+}
+
+export const pageCompileAll = () => {
+ // Read pages
+ Object.entries({
+ 'index': 'home',
+ 'privacy': 'privacy'
+ }).forEach(entry => {
+ const [ out, page ] = entry;
+ pageGenerate(out, page);
+ });
+}
diff --git a/src/pages/home/home.css b/src/pages/home/home.css
new file mode 100644
index 0000000..e69de29
diff --git a/src/pages/home/home.html b/src/pages/home/home.html
new file mode 100644
index 0000000..7893ade
--- /dev/null
+++ b/src/pages/home/home.html
@@ -0,0 +1,107 @@
+
+ Dominic Masters
+ Developer, Nerd, Occasionally Funny.
+
+
+ Australian nerd with a passion for all things computing, coffee and
+ video games. Currently residing in Sydney, Australia.
+
+
+
+ Programming since before the internet was cool.
+
+
+
+
+
+
+ Programming Technical Lead
+
+ I am a programmer born and bred. Developing since I was old enough to
+ type on a keyboard and continue to refine my skills more and more
+ every day. Backed by 14 years of experience and countless lines of code
+ committed.
+
+
+
+ Background with more anagrams than a bureaucrat, here are some of the
+ highlights only;
+
+ eCommerce ( Shopify, Magento, WooCommerce, neto )
+ Traditional Web ( HTML, CSS, JS )
+ Modern Web ( ES6, TS, GraphQL )
+ Server Side ( NodeJS, PHP, .NET )
+ Client Side ( C/C++, Java )
+ Database Side ( SQL, NoSQL )
+ Frameworks ( React, Web Components )
+ Antiques ( Lua, jQuery, SOAPXML )
+ Flashy ( OpenGL, Vulkan, Unity )
+ Boring ( Google Data Studio )
+ Servers ( AWS, CloudFlare, Heroku, Azure )
+ Every "API"
+ Ask me for a comprehensive list.
+
+
+
+
+
+
+
+
+ Shopify Expert
+
+ I'm currently working full-time as a Technical Lead working with the
+ Shopify Plus platform. I have been working with the platform
+ every day for over 4 years, and enjoy working with it immensely.
+
+
+
+ Working with Liquid, REST and GraphQL App Development and general
+ Shopify consulting. I have had the privilage of working with countless
+ Shopify Plus clients.
+
+
+
+ Despite Shopify's reputation, I have been able to make it do tricks
+ thought impossible. I find working with Shopify's unique quirks very
+ rewarding, and continuously find ways to surprise everyone, including
+ myself.
+
+
+
+
+
+
+ Personal
+
+ I spend most of my spare time considering programming concepts, and
+ take a very pragmatic approach to planning software development. I
+ enjoy unpacking what programming itself means and entails to come up
+ with new paradigms to use in my job.
+
+
+
+ I also have an appreciation for the raw power of technology and making
+ something beautiful and intricate out of the minimal amount of tools.
+ Hence this website appearing so basic.
+
+
+
+ I also enjoy video games and antique electronics. I still own four old
+ CRT Monitors that I love and use as my primary video game displays.
+
+
+
+
+
+
+ Social
+
+ I don't do social media often, however you can find me below.
+ For business enquiries contact I suggest LinkedIn.
+
+
+ LinkedIn
+ GitHub
+ Twitter
+
\ No newline at end of file
diff --git a/src/pages/privacy/privacy.html b/src/pages/privacy/privacy.html
new file mode 100644
index 0000000..2f0a0a4
--- /dev/null
+++ b/src/pages/privacy/privacy.html
@@ -0,0 +1,175 @@
+Privacy Policy
+Effective date: June 27, 2018
+
+ domsPlace ("us", "we", or "our") operates
+ the https://domsplace.com website (the
+ "Service").
+
+
+
+ This page informs you of our policies regarding the collection, use, and
+ disclosure of personal data when you use our Service and the choices you
+ have associated with that data.
+
+
+
+ We use your data to provide and improve the Service. By using the
+ Service, you agree to the collection and use of information in
+ accordance with this policy. Unless otherwise defined in this Privacy
+ Policy, terms used in this Privacy Policy have the same meanings as in
+ our Terms and Conditions, accessible from
+ https://domsplace.com
+
+
+
+
+
+
+
+Information Collection And Use
+
+ We collect several different types of information for various purposes
+ to provide and improve our Service to you.
+
+
+Types of Data Collected
+Personal Data
+
+ While using our Service, we may ask you to provide us with certain
+ personally identifiable information that can be used to contact or
+ identify you ("Personal Data"). Personally identifiable information may
+ include, but is not limited to:
+
+
+ Email address
+ First name and last name
+ Phone number
+ Cookies and Usage Data
+
+
+Usage Data
+
+ We may also collect information how the Service is accessed and used =
+ ("Usage Data"). This Usage Data may include information such as your
+ computer's Internet Protocol address (e.g. IP address), browser type,
+ browser version, the pages of our Service that you visit, the time and
+ date of your visit, the time spent on those pages, unique device
+ identifiers and other diagnostic data.
+
+
+Tracking & Cookies Data
+
+ We use cookies and similar tracking technologies to track the activity
+ on our Service and hold certain information.
+
+
+ Cookies are files with small amount of data which may include an
+ anonymous unique identifier. Cookies are sent to your browser from a
+ website and stored on your device. Tracking technologies also used are
+ beacons, tags, and scripts to collect and track information and to
+ improve and analyze our Service.
+
+
+ You can instruct your browser to refuse all cookies or to indicate when
+ a cookie is being sent. However, if you do not accept cookies, you may
+ not be able to use some portions of our Service.
+
+Examples of Cookies we use:
+
+ Session Cookies. We use Session Cookies to operate our Service.
+ Preference Cookies. We use Preference Cookies to remember your preferences and various settings.
+ Security Cookies. We use Security Cookies for security purposes.
+
+
+
+
+
+
+Use of Data
+domsPlace uses the collected data for various purposes:
+
+ To provide and maintain the Service
+ To notify you about changes to our Service
+ To allow you to participate in interactive features of our Service when you choose to do so
+ To provide customer care and support
+ To provide analysis or valuable information so that we can improve the Service
+ To monitor the usage of the Service
+ To detect, prevent and address technical issues
+
+
+
+
+
+
+Transfer Of Data
+Your information, including Personal Data, may be transferred to — and maintained on — computers located outside of your state, province, country or other governmental jurisdiction where the data protection laws may differ than those from your jurisdiction.
+If you are located outside Australia and choose to provide information to us, please note that we transfer the data, including Personal Data, to Australia and process it there.
+Your consent to this Privacy Policy followed by your submission of such information represents your agreement to that transfer.
+domsPlace will take all steps reasonably necessary to ensure that your data is treated securely and in accordance with this Privacy Policy and no transfer of your Personal Data will take place to an organization or a country unless there are adequate controls in place including the security of your data and other personal information.
+
+
+
+
+
+Disclosure Of Data
+Legal Requirements
+domsPlace may disclose your Personal Data in the good faith belief that such action is necessary to:
+
+ To comply with a legal obligation
+ To protect and defend the rights or property of domsPlace
+ To prevent or investigate possible wrongdoing in connection with the Service
+ To protect the personal safety of users of the Service or the public
+ To protect against legal liability
+
+
+
+
+
+
+Security Of Data
+The security of your data is important to us, but remember that no method of transmission over the Internet, or method of electronic storage is 100% secure. While we strive to use commercially acceptable means to protect your Personal Data, we cannot guarantee its absolute security.
+
+
+
+
+
+Service Providers
+We may employ third party companies and individuals to facilitate our Service ("Service Providers"), to provide the Service on our behalf, to perform Service-related services or to assist us in analyzing how our Service is used.
+These third parties have access to your Personal Data only to perform these tasks on our behalf and are obligated not to disclose or use it for any other purpose.
+
+Analytics
+We may use third-party Service Providers to monitor and analyze the use of our Service.
+
+
+ Google Analytics
+ Google Analytics is a web analytics service offered by Google that tracks and reports website traffic. Google uses the data collected to track and monitor the use of our Service. This data is shared with other Google services. Google may use the collected data to contextualize and personalize the ads of its own advertising network.
+ You can opt-out of having made your activity on the Service available to Google Analytics by installing the Google Analytics opt-out browser add-on. The add-on prevents the Google Analytics JavaScript (ga.js, analytics.js, and dc.js) from sharing information with Google Analytics about visits activity.
+ For more information on the privacy practices of Google, please visit the Google Privacy & Terms web page: https://policies.google.com/privacy?hl=en
+
+
+
+
+
+
+
+
+Links To Other Sites
+Our Service may contain links to other sites that are not operated by us. If you click on a third party link, you will be directed to that third party's site. We strongly advise you to review the Privacy Policy of every site you visit.
+We have no control over and assume no responsibility for the content, privacy policies or practices of any third party sites or services.
+
+
+
+
+
+Children's Privacy
+Our Service does not address anyone under the age of 18 ("Children").
+We do not knowingly collect personally identifiable information from anyone under the age of 18. If you are a parent or guardian and you are aware that your Children has provided us with Personal Data, please contact us. If we become aware that we have collected Personal Data from children without verification of parental consent, we take steps to remove that information from our servers.
+
+
+
+
+
+Changes To This Privacy Policy
+We may update our Privacy Policy from time to time. We will notify you of any changes by posting the new Privacy Policy on this page.
+We will let you know via email and/or a prominent notice on our Service, prior to the change becoming effective and update the "effective date" at the top of this Privacy Policy.
+You are advised to review this Privacy Policy periodically for any changes. Changes to this Privacy Policy are effective when they are posted on this page.
\ No newline at end of file
diff --git a/src/partials/layout/layout.css b/src/partials/layout/layout.css
new file mode 100644
index 0000000..be145d3
--- /dev/null
+++ b/src/partials/layout/layout.css
@@ -0,0 +1,69 @@
+/* Elements */
+* {
+ box-sizing: border-box;
+}
+
+html,body {
+ margin: 0;
+ padding: 0;
+ font-size: 16px;
+ background: #000;
+ color: #FFF;
+ font-family: monospace;
+}
+
+a { color: #00F; text-decoration: underline; }
+a:visited,a:active { color: #C0F; }
+a:hover { text-decoration: none; }
+
+
+/* Objects */
+.o-break {
+ padding: 3rem 0;
+}
+
+
+/* Components */
+.c-header {
+ text-align: center;
+ padding: 4rem 1rem;
+ font-weight: bold;
+ max-width: 800px;
+ margin: auto;
+}
+
+.c-header__title {
+ font-size: 1.5rem;
+}
+
+.c-header__title-link {
+ color: inherit;
+ text-decoration: none;
+}
+.c-header__title-link:visited,.c-header__title-link:active {
+ color: inherit;
+}
+
+.c-header__subtitle {
+ font-style: italic;
+}
+
+
+.c-main {
+ max-width: 800px;
+ margin: auto;
+ padding: 0 1em;
+}
+
+
+.c-footer {
+ text-align: center;
+ padding: 12rem 1rem 10rem;
+ font-weight: bold;
+ max-width: 800px;
+ margin: auto;
+}
+
+.c-footer__link + .c-footer__link {
+ margin-left: 0.5rem;
+}
\ No newline at end of file
diff --git a/src/partials/layout/layout.html b/src/partials/layout/layout.html
new file mode 100644
index 0000000..ce2e5fd
--- /dev/null
+++ b/src/partials/layout/layout.html
@@ -0,0 +1,45 @@
+
+
+
+
+
+
+ domsPlace, Personal Website of Technical Lead Dominic Masters
+
+
+
+
+
+
+
+
+
+
+ {{main}}
+
+
+
+
+
\ No newline at end of file
diff --git a/src/private/.gitignore b/src/private/.gitignore
deleted file mode 100644
index f770a93..0000000
--- a/src/private/.gitignore
+++ /dev/null
@@ -1,3 +0,0 @@
-dist/
-*.log
-*.lock
\ No newline at end of file
diff --git a/src/private/jest.config.js b/src/private/jest.config.js
deleted file mode 100644
index af3e56e..0000000
--- a/src/private/jest.config.js
+++ /dev/null
@@ -1,14 +0,0 @@
-module.exports = {
- "roots": [
- "/src"
- ],
- "transform": {
- "^.+\\.ts?$": "ts-jest"
- },
- "testRegex": "(/__tests__/.*|(\\.|/)(test|spec))\\.ts?$",
- "moduleFileExtensions": [
- "ts",
- "js",
- "json"
- ]
-}
diff --git a/src/private/package.json b/src/private/package.json
deleted file mode 100644
index f438fa1..0000000
--- a/src/private/package.json
+++ /dev/null
@@ -1,51 +0,0 @@
-{
- "name": "domsplace-backend",
- "version": "7.0.0",
- "description": "Personal website for Dominic \"YouWish\" Masters.",
- "main": "./dist/private/",
- "scripts": {
- "build": "tsc -p .",
- "start:prod": "cross-env NODE_ENV=production \"serverless offline\"",
- "start:dev": "cross-env NODE_ENV=development \"serverless offline --port 3001\"",
- "start": "npm run start:prod",
- "watch": "npm run start:dev",
- "deploy": "serverless deploy",
- "test": "yarn jest"
- },
- "repository": {
- "type": "git",
- "url": "git+https://github.com/YourWishes/domsPlace.git"
- },
- "keywords": [
- "domsplace",
- "personal",
- "portfolio",
- "website"
- ],
- "author": "Dominic Masters",
- "license": "MIT",
- "bugs": {
- "url": "https://github.com/YourWishes/domsPlace/issues"
- },
- "homepage": "https://github.com/YourWishes/domsPlace#readme",
- "dependencies": {
- "email-validator": "^2.0.4",
- "nodemailer": "^6.3.0",
- "serverless-plugin-include-dependencies": "^3.2.1"
- },
- "devDependencies": {
- "@types/jest": "^25.1.1",
- "@types/node": "^12.7.1",
- "@types/nodemailer": "^6.2.1",
- "cross-env": "^5.2.0",
- "escape-html": "^1.0.3",
- "jest": "^24.9.0",
- "serverless": "^1.51.0",
- "serverless-finch": "^2.4.3",
- "serverless-offline": "^5.11.0",
- "ts-jest": "^24.1.0",
- "ts-node": "^8.3.0",
- "typescript": "^3.5.3",
- "utility-types": "^3.7.0"
- }
-}
diff --git a/src/private/serverless.yml b/src/private/serverless.yml
deleted file mode 100644
index 31bfe21..0000000
--- a/src/private/serverless.yml
+++ /dev/null
@@ -1,53 +0,0 @@
-org: yourwishes
-service: domsplace
-
-frameworkVersion: ">=1.26.0"
-
-package:
- excludeDevDependencies: false
- individually: true
- include:
- - dist/**
-
-provider:
- name: aws
- runtime: nodejs10.x
- stage: ${opt:stage, "prod"}
- region: ap-southeast-2
- memorySize: 512
- deploymentBucket:
- name: domsplace-${self:provider.stage}-${self:provider.region}-private
- environment:
- EMAIL_HOST: ${self:custom.variables.email.host}
- EMAIL_PORT: ${self:custom.variables.email.port}
- EMAIL_USER: ${self:custom.variables.email.user}
- EMAIL_PASS: ${self:custom.variables.email.pass}
- EMAIL_DEST: ${self:custom.variables.email.dest}
-
-functions:
- ping:
- handler: dist/functions/ping/ping.ping
- events:
- - http:
- method: GET
- path: ping
- cors: true
- sendMail:
- handler: dist/functions/mail/send.sendMail
- events:
- - http:
- method: POST
- path: mail/send
- cors: true
-
-plugins:
- - serverless-plugin-include-dependencies
- - serverless-offline
-
-custom:
- ssm: '/aws/reference/secretsmanager/prod.domsPlace.'
- serverless-offline:
- disableCookieValidation: true
- port: 3001
- variables:
- email: ${ssm:${self:custom.ssm}email~true}
\ No newline at end of file
diff --git a/src/private/src/functions/mail/__tests__/send.test.ts b/src/private/src/functions/mail/__tests__/send.test.ts
deleted file mode 100644
index 37a2a39..0000000
--- a/src/private/src/functions/mail/__tests__/send.test.ts
+++ /dev/null
@@ -1,8 +0,0 @@
-import { sendMail } from './../send';
-
-describe('sendMail', () => {
- it('should require a body', async () => {
- await expect(sendMail({ body: null })).resolves.toHaveProperty('statusCode', 400);
- });
-
-})
\ No newline at end of file
diff --git a/src/private/src/functions/mail/send.ts b/src/private/src/functions/mail/send.ts
deleted file mode 100644
index 0b83717..0000000
--- a/src/private/src/functions/mail/send.ts
+++ /dev/null
@@ -1,56 +0,0 @@
-import { withHandler } from "../../handler/handler";
-import { validate } from 'email-validator';
-import { createTransport } from 'nodemailer';
-import * as escapeHTML from 'escape-html';
-
-export interface sendMailParams {
- name:string;
- email:string;
- message:string;
-}
-
-export const sendMail = withHandler(async (e,c) => {
- if(!e.body) return { statusCode: 400, body: 'Missing Contact Details' };
-
- let { name, email, message } = e.body;
- if(!name) return { statusCode: 400, body: 'Missing Contact Name' };
- if(!email) return { statusCode: 400, body: 'Missing Contact Email' };
- if(!message) return { statusCode: 400, body: 'Missing Contact Message' };
-
- //Validate
- if(name.length > 128 || !name.replace(/\s/g, '').length) return { statusCode: 400, body: 'Invalid Name' };
- if(!validate(email)) return { statusCode: 400, body: 'Invalid Email' };
- if(message.length > 10000 || !message.replace(/\s/g,'').length) {
- return { statusCode: 400, body: 'Invalid Messatge' };
- }
-
- //Prepare mail
- let {
- EMAIL_HOST, EMAIL_PORT, EMAIL_USER, EMAIL_PASS, EMAIL_DEST, EMAIL_FROM
- } = process.env
-
- let transporter = createTransport({
- host: EMAIL_HOST,
- port: parseInt(EMAIL_PORT),
- secure: true,
- auth: {
- user: EMAIL_USER, pass: EMAIL_PASS
- }
- });
-
- let x = await transporter.sendMail({
- from: `${name} <${email}>`,
- to: EMAIL_DEST,
- subject: 'Contact Message Received',
- text: `Contact Message Received:\n${message}\nFrom: ${name} ${email}`,
- html: `
- Contact Message Received
- You have received a contact message from ${escapeHTML(name)} - ${escapeHTML(email)} who wrote:
-
- ${escapeHTML(message)}
-
- Time: ${new Date().toLocaleString()}
- `
- });
- return { statusCode: 200, body: x && x.accepted && x.accepted.length ? true : false }
-});
\ No newline at end of file
diff --git a/src/private/src/functions/ping/__tests__/ping.test.ts b/src/private/src/functions/ping/__tests__/ping.test.ts
deleted file mode 100644
index e143b7a..0000000
--- a/src/private/src/functions/ping/__tests__/ping.test.ts
+++ /dev/null
@@ -1,8 +0,0 @@
-import { ping } from './../ping';
-
-describe('ping', () => {
- it('shold return a hello world and a 200 code', async () => {
- await expect(ping()).resolves.toHaveProperty('body', 'Thank funk!');
- await expect(ping()).resolves.toHaveProperty('statusCode', 200);
- })
-})
\ No newline at end of file
diff --git a/src/private/src/functions/ping/ping.ts b/src/private/src/functions/ping/ping.ts
deleted file mode 100644
index d95417c..0000000
--- a/src/private/src/functions/ping/ping.ts
+++ /dev/null
@@ -1,5 +0,0 @@
-import { withHandler } from "../../handler/handler";
-
-export const ping = withHandler(async () => {
- return { statusCode: 200, body: 'Thank funk!' };
-});
\ No newline at end of file
diff --git a/src/private/src/handler/__tests__/handler.test.ts b/src/private/src/handler/__tests__/handler.test.ts
deleted file mode 100644
index a2fcd41..0000000
--- a/src/private/src/handler/__tests__/handler.test.ts
+++ /dev/null
@@ -1,43 +0,0 @@
-import { withHandler, APICallable, DEFAULT_HEADERS } from './../handler';
-
-describe('withHandler', () => {
- it('should wrap an async function into a serverless callbackable function', () => {
- let callbackable = jest.fn(async () => ({ body: 'Hello World', statusCode: 200 }));
- let x = withHandler(callbackable);
- expect(typeof x).toBe('function');
- });
-
- it('should call the promise and pass the result to the callback', async () => {
- let callbackable = async () => ({ body: 'Hello World', statusCode: 200 });
- let x = withHandler(callbackable);
- let fn = jest.fn();
-
- x({} as any, null, fn);
-
- await new Promise(resolve => setImmediate(resolve));
-
- expect(fn).toHaveBeenCalled();
- });
-
- it('should set the content type', async () => {
- let x = withHandler(async () => ({ body: 'Hello World', statusCode: 200 }));
- let fn = jest.fn();
-
- x({} as any, null, fn);
-
- await new Promise(resolve => setImmediate(resolve));
-
- expect(fn).toHaveBeenCalledWith(null, { body: '"Hello World"', statusCode: 200,
- headers: DEFAULT_HEADERS
- });
- });
-
- it('should return the invoked functions returned promise', async () => {
- let x = withHandler(async () => ({ body: 'Hello World', statusCode: 200 }));
- let fn = jest.fn();
- let prom = x({} as any, null, fn);
-
- let result = await prom;
- expect(result).toStrictEqual({ body: 'Hello World', statusCode: 200 });
- })
-})
\ No newline at end of file
diff --git a/src/private/src/handler/handler.ts b/src/private/src/handler/handler.ts
deleted file mode 100644
index 04a99c8..0000000
--- a/src/private/src/handler/handler.ts
+++ /dev/null
@@ -1,71 +0,0 @@
-export const DEFAULT_HEADERS = {
- 'Content-Type': 'application/json',
- 'Access-Control-Allow-Origin': '*',
- 'Access-Control-Allow-Credentials': true
-}
-
-export type APIResponse = {
- statusCode?:number;
- body:any;
- headers?:{
- [key:string]:string,
- "Content-Type"?:string
- };
-}
-
-export type APICallable = (event:APIEvent, context:any) => Promise;
-
-export type APIMethod = 'GET'|'POST'|'PUT'|'PATCH'|'DELETE'|'TRACE'|'OPTIONS'|'CONNECT';
-
-export interface APIEvent {
- body:T;
- headers?:{[key:string]:string};
- httpMethod?:APIMethod;
- isOffline?:boolean;
- multiValueHeaders?:{[key:string]:string};
- multiValueQueryStringParameters?:{[key:string]:string};
- path?:string;
- pathParameters?:never;
- queryStringParameters?:{[key:string]:string};
- requestContext?:{[key:string]:string};
- resource?:string;
- stageVariables?:any;
-}
-
-export const withHandler = (callable:APICallable) => {
- return (event?:APIEvent, context?, callback?) => {
- try {
- if(!event) throw new Error();
- if(event.body) event.body = JSON.parse(event.body as any) as T;
- } catch(e) {
- console.error(e);
- callback(null, {
- statusCode: 400, headers: DEFAULT_HEADERS,
- body: JSON.stringify('Invalid body')
- });
- }
-
- return callable(event, context).then(d => {
- if(!callback) return d;
- let contentType = (d.headers?d.headers['Content-Type']:null) || 'application/json';
- let json = contentType.indexOf('application/json') !== -1;
-
- callback(null, {
- ...d,
- body: json ? JSON.stringify(d.body) : d.body,
- statusCode: d.statusCode || 200,
- headers: {
- ...DEFAULT_HEADERS,
- ...(d.headers||{})
- }
- });
-
- return d;
- }).catch(ex => {
- if(callback) {
- callback(null, { statusCode: 500, body: null, headers: DEFAULT_HEADERS });
- }
- throw ex;
- })
- };
-}
\ No newline at end of file
diff --git a/src/private/tsconfig.json b/src/private/tsconfig.json
deleted file mode 100644
index 4764d7d..0000000
--- a/src/private/tsconfig.json
+++ /dev/null
@@ -1,14 +0,0 @@
-{
- "compilerOptions": {
- "outDir": "./dist",
- "target": "es5",
- "module": "commonjs",
- "noImplicitAny": false,
- "sourceMap": true,
- "baseUrl": "./src/"
- },
- "include": [
- "./src/**/*.ts"
- ],
- "exclude": ["node_modules", "**/*.test.ts"]
-}
diff --git a/src/public/.gitignore b/src/public/.gitignore
deleted file mode 100644
index 56af776..0000000
--- a/src/public/.gitignore
+++ /dev/null
@@ -1,2 +0,0 @@
-public/
-*.lock
diff --git a/src/public/gatsby-browser.js b/src/public/gatsby-browser.js
deleted file mode 100644
index 55a9428..0000000
--- a/src/public/gatsby-browser.js
+++ /dev/null
@@ -1 +0,0 @@
-import 'normalize.css';
\ No newline at end of file
diff --git a/src/public/gatsby-config.js b/src/public/gatsby-config.js
deleted file mode 100644
index 19c17e9..0000000
--- a/src/public/gatsby-config.js
+++ /dev/null
@@ -1,49 +0,0 @@
-const path = require('path');
-const TSConfig = require('./tsconfig.json');
-
-module.exports = {
- plugins: [
- {
- resolve: `gatsby-source-filesystem`,
- options: {
- path: path.join(__dirname, 'src', 'assets', 'images'),
- name: 'images'
- }
- },
-
- 'gatsby-plugin-typescript',
- 'gatsby-plugin-react-helmet',
- 'gatsby-plugin-styled-components',
- 'gatsby-transformer-sharp',
- 'gatsby-plugin-sharp',
-
- {
- resolve: 'gatsby-plugin-alias-imports',
- options: {
- alias: Object.entries(TSConfig.compilerOptions.paths).reduce((x, [key, value]) => {
- let k = key.split('/').filter(f => f && f != '*').join('/');
- let v = value.find(v => v).split('/').filter(f => f && f != '*');
- return { ...x, [k]: path.resolve(__dirname, TSConfig.compilerOptions.baseUrl, ...v) };
- }, {})
- }
- },
-
- {
- resolve: 'gatsby-plugin-google-fonts',
- options: {
- fonts: [
- 'Bitter',
- 'Nanum Gothic'
- ],
- display: 'swap'
- }
- },
-
- {
- resolve: `gatsby-plugin-layout`,
- options: {
- component: require.resolve(`./src/layouts/main.tsx`),
- },
- }
- ]
-}
diff --git a/src/public/package.json b/src/public/package.json
deleted file mode 100644
index 7d921eb..0000000
--- a/src/public/package.json
+++ /dev/null
@@ -1,71 +0,0 @@
-{
- "name": "domsplace-frontend",
- "version": "8.0.0",
- "description": "Personal website for Dominic \"YourWishes\" Masters.",
- "scripts": {
- "build": "gatsby build",
- "develop": "gatsby develop",
- "start": "npm run develop",
- "serve": "gatsby serve",
- "deploy": "serverless client deploy --no-confirm",
- "clean": "gatsby clean",
- "test": "echo \"No tests defined\""
- },
- "repository": {
- "type": "git",
- "url": "git+https://github.com/YourWishes/domsPlaceNew.git"
- },
- "keywords": [
- "domsplace",
- "personal",
- "portfolio",
- "website"
- ],
- "author": "Dominic Masters",
- "license": "MIT",
- "bugs": {
- "url": "https://github.com/YourWishes/domsPlaceNew/issues"
- },
- "homepage": "https://domsplace.com",
- "dependencies": {
- "babel-plugin-styled-components": "^1.10.7",
- "gatsby": "^2.18.12",
- "gatsby-image": "^2.2.39",
- "gatsby-plugin-alias-imports": "^1.0.5",
- "gatsby-plugin-google-fonts": "^1.0.1",
- "gatsby-plugin-layout": "^1.1.22",
- "gatsby-plugin-react-helmet": "^3.1.21",
- "gatsby-plugin-sharp": "^2.4.3",
- "gatsby-plugin-styled-components": "^3.1.18",
- "gatsby-plugin-typescript": "^2.1.26",
- "gatsby-source-filesystem": "^2.1.46",
- "gatsby-transformer-sharp": "^2.3.13",
- "normalize.css": "^8.0.1",
- "react": "^16.12.0",
- "react-dom": "^16.12.0",
- "react-helmet": "^5.2.1",
- "react-hook-form": "^4.8.0",
- "react-syntax-highlighter": "^12.2.1",
- "react-transition-group": "^4.3.0",
- "styled-components": "^5.0.0",
- "yup": "^0.28.1"
- },
- "devDependencies": {
- "@types/node": "^13.5.0",
- "@types/react": "^16.9.19",
- "@types/react-dom": "^16.9.5",
- "@types/react-helmet": "^5.0.15",
- "@types/react-transition-group": "^4.2.3",
- "@types/styled-components": "^4.4.2",
- "@types/yup": "^0.26.29",
- "serverless": "^1.63.0",
- "serverless-finch": "^2.5.2",
- "serverless-plugin-include-dependencies": "^4.0.1"
- },
- "browserslist": [
- ">0.2%",
- "not dead",
- "not ie <= 11",
- "not op_mini all"
- ]
-}
diff --git a/src/public/serverless.yml b/src/public/serverless.yml
deleted file mode 100644
index d88e129..0000000
--- a/src/public/serverless.yml
+++ /dev/null
@@ -1,28 +0,0 @@
-org: yourwishes
-service: domsplace
-
-frameworkVersion: ">=1.26.0"
-
-provider:
- name: aws
- runtime: nodejs10.x
- stage: ${opt:stage, "prod"}
- region: ap-southeast-2
- memorySize: 512
-
-package:
- excludeDevDependencies: false
- individually: true
- include:
- - public/**
-
-plugins:
- - serverless-plugin-include-dependencies
- - serverless-finch
-
-custom:
- client:
- bucketName: domsplace-${self:provider.stage}-${self:provider.region}-public
- distributionFolder: public/
- indexDocument: index.html
- errorDocument: index.html
\ No newline at end of file
diff --git a/src/public/src/api/APIRequest.tsx b/src/public/src/api/APIRequest.tsx
deleted file mode 100644
index fbff0a6..0000000
--- a/src/public/src/api/APIRequest.tsx
+++ /dev/null
@@ -1,11 +0,0 @@
-export const APIRequest = (url:string, body?:object) => {
- return fetch(`https://api.domsplace.com/v1/${url}`, {
- method: body ? 'POST' : 'GET',
- body: body ? JSON.stringify(body) : null,
- mode: 'no-cors',
- headers: {
- 'Content-Type': 'application/json',
- 'Accept': 'application/json'
- }
- }).then(e => e.json());
-}
\ No newline at end of file
diff --git a/src/public/src/api/SendMail.tsx b/src/public/src/api/SendMail.tsx
deleted file mode 100644
index e2e38f8..0000000
--- a/src/public/src/api/SendMail.tsx
+++ /dev/null
@@ -1,7 +0,0 @@
-import { APIRequest } from "./APIRequest";
-
-export interface SendMailParams {
- name:string, email:string, message:string
-};
-
-export const sendMail = (data:SendMailParams) => APIRequest('mail/send', data).then(e => e.body);
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/cpp/cpp-logo.svg b/src/public/src/assets/images/branding/cpp/cpp-logo.svg
deleted file mode 100644
index 5e11707..0000000
--- a/src/public/src/assets/images/branding/cpp/cpp-logo.svg
+++ /dev/null
@@ -1,25 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/csharp/csharp-logo.svg b/src/public/src/assets/images/branding/csharp/csharp-logo.svg
deleted file mode 100644
index ddc79cd..0000000
--- a/src/public/src/assets/images/branding/csharp/csharp-logo.svg
+++ /dev/null
@@ -1,32 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/digitalocean/digitalocean-logo.svg b/src/public/src/assets/images/branding/digitalocean/digitalocean-logo.svg
deleted file mode 100644
index 0026c1b..0000000
--- a/src/public/src/assets/images/branding/digitalocean/digitalocean-logo.svg
+++ /dev/null
@@ -1,40 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/discord/discord-logo.svg b/src/public/src/assets/images/branding/discord/discord-logo.svg
deleted file mode 100644
index 65c9fcc..0000000
--- a/src/public/src/assets/images/branding/discord/discord-logo.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/google-cloud/google-cloud-logo.svg b/src/public/src/assets/images/branding/google-cloud/google-cloud-logo.svg
deleted file mode 100644
index 6fd725f..0000000
--- a/src/public/src/assets/images/branding/google-cloud/google-cloud-logo.svg
+++ /dev/null
@@ -1,12 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/graphql/graphql-logo.svg b/src/public/src/assets/images/branding/graphql/graphql-logo.svg
deleted file mode 100644
index 14082a5..0000000
--- a/src/public/src/assets/images/branding/graphql/graphql-logo.svg
+++ /dev/null
@@ -1,64 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/heroku/heroku-logo.svg b/src/public/src/assets/images/branding/heroku/heroku-logo.svg
deleted file mode 100644
index c287cd8..0000000
--- a/src/public/src/assets/images/branding/heroku/heroku-logo.svg
+++ /dev/null
@@ -1,18 +0,0 @@
-
-
-
-
-
- image/svg+xml
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/java/java-logo.svg b/src/public/src/assets/images/branding/java/java-logo.svg
deleted file mode 100644
index 28d4a95..0000000
--- a/src/public/src/assets/images/branding/java/java-logo.svg
+++ /dev/null
@@ -1,10 +0,0 @@
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/jquery/jquery-logo.svg b/src/public/src/assets/images/branding/jquery/jquery-logo.svg
deleted file mode 100644
index 621989a..0000000
--- a/src/public/src/assets/images/branding/jquery/jquery-logo.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/lwjgl/lwjgl-logo.svg b/src/public/src/assets/images/branding/lwjgl/lwjgl-logo.svg
deleted file mode 100644
index 3410765..0000000
--- a/src/public/src/assets/images/branding/lwjgl/lwjgl-logo.svg
+++ /dev/null
@@ -1,32 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/monogame/monogame-logo.svg b/src/public/src/assets/images/branding/monogame/monogame-logo.svg
deleted file mode 100644
index 226a1d2..0000000
--- a/src/public/src/assets/images/branding/monogame/monogame-logo.svg
+++ /dev/null
@@ -1,30 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/mysql/mysql-logo.svg b/src/public/src/assets/images/branding/mysql/mysql-logo.svg
deleted file mode 100644
index 98ebde9..0000000
--- a/src/public/src/assets/images/branding/mysql/mysql-logo.svg
+++ /dev/null
@@ -1,47 +0,0 @@
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/neto/neto-logo.svg b/src/public/src/assets/images/branding/neto/neto-logo.svg
deleted file mode 100644
index fad1b20..0000000
--- a/src/public/src/assets/images/branding/neto/neto-logo.svg
+++ /dev/null
@@ -1,13 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/nodejs/nodejs-logo.svg b/src/public/src/assets/images/branding/nodejs/nodejs-logo.svg
deleted file mode 100644
index b51f302..0000000
--- a/src/public/src/assets/images/branding/nodejs/nodejs-logo.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/opengl/opengl-logo.svg b/src/public/src/assets/images/branding/opengl/opengl-logo.svg
deleted file mode 100644
index 84c5d12..0000000
--- a/src/public/src/assets/images/branding/opengl/opengl-logo.svg
+++ /dev/null
@@ -1,18 +0,0 @@
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/pgsql/pgsql-logo.svg b/src/public/src/assets/images/branding/pgsql/pgsql-logo.svg
deleted file mode 100644
index 8666f75..0000000
--- a/src/public/src/assets/images/branding/pgsql/pgsql-logo.svg
+++ /dev/null
@@ -1,20 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/php/php-logo.svg b/src/public/src/assets/images/branding/php/php-logo.svg
deleted file mode 100644
index e4f137c..0000000
--- a/src/public/src/assets/images/branding/php/php-logo.svg
+++ /dev/null
@@ -1,96 +0,0 @@
-
-
- Official PHP Logo
-
-
-
- image/svg+xml
-
- Official PHP Logo
-
-
- Colin Viebrock
-
-
-
-
-
-
-
-
-
-
-
- Copyright Colin Viebrock 1997 - All rights reserved.
-
-
- 1997
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/react/react-logo.svg b/src/public/src/assets/images/branding/react/react-logo.svg
deleted file mode 100644
index b3887f3..0000000
--- a/src/public/src/assets/images/branding/react/react-logo.svg
+++ /dev/null
@@ -1,37 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/redux/redux-logo.svg b/src/public/src/assets/images/branding/redux/redux-logo.svg
deleted file mode 100644
index 62f4d5c..0000000
--- a/src/public/src/assets/images/branding/redux/redux-logo.svg
+++ /dev/null
@@ -1,22 +0,0 @@
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/shopify/shopify-logo.svg b/src/public/src/assets/images/branding/shopify/shopify-logo.svg
deleted file mode 100644
index 7c88138..0000000
--- a/src/public/src/assets/images/branding/shopify/shopify-logo.svg
+++ /dev/null
@@ -1,23 +0,0 @@
-
-
-
-
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/twitch/twitch-logo.svg b/src/public/src/assets/images/branding/twitch/twitch-logo.svg
deleted file mode 100644
index 905e364..0000000
--- a/src/public/src/assets/images/branding/twitch/twitch-logo.svg
+++ /dev/null
@@ -1 +0,0 @@
-Glitch
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/twitter/twitter-logo.svg b/src/public/src/assets/images/branding/twitter/twitter-logo.svg
deleted file mode 100644
index 2832e7b..0000000
--- a/src/public/src/assets/images/branding/twitter/twitter-logo.svg
+++ /dev/null
@@ -1 +0,0 @@
-Twitter_Logo_Blue
\ No newline at end of file
diff --git a/src/public/src/assets/images/branding/typescript/typescript-logo.svg b/src/public/src/assets/images/branding/typescript/typescript-logo.svg
deleted file mode 100644
index bbf2809..0000000
--- a/src/public/src/assets/images/branding/typescript/typescript-logo.svg
+++ /dev/null
@@ -1,40 +0,0 @@
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/unity/unity-logo.svg b/src/public/src/assets/images/branding/unity/unity-logo.svg
deleted file mode 100644
index 88622dd..0000000
--- a/src/public/src/assets/images/branding/unity/unity-logo.svg
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-
-
-
-
diff --git a/src/public/src/assets/images/branding/webpack/webpack-logo.svg b/src/public/src/assets/images/branding/webpack/webpack-logo.svg
deleted file mode 100644
index 8f7eaa5..0000000
--- a/src/public/src/assets/images/branding/webpack/webpack-logo.svg
+++ /dev/null
@@ -1 +0,0 @@
-icon
\ No newline at end of file
diff --git a/src/public/src/assets/images/people/dominic/head.png b/src/public/src/assets/images/people/dominic/head.png
deleted file mode 100644
index 7451356..0000000
Binary files a/src/public/src/assets/images/people/dominic/head.png and /dev/null differ
diff --git a/src/public/src/assets/images/websites/bundlfresh.jpg b/src/public/src/assets/images/websites/bundlfresh.jpg
deleted file mode 100644
index 407ee02..0000000
Binary files a/src/public/src/assets/images/websites/bundlfresh.jpg and /dev/null differ
diff --git a/src/public/src/assets/images/websites/cocksox.jpg b/src/public/src/assets/images/websites/cocksox.jpg
deleted file mode 100644
index a9d4e5d..0000000
Binary files a/src/public/src/assets/images/websites/cocksox.jpg and /dev/null differ
diff --git a/src/public/src/assets/images/websites/domsPlace.jpg b/src/public/src/assets/images/websites/domsPlace.jpg
deleted file mode 100644
index e568b28..0000000
Binary files a/src/public/src/assets/images/websites/domsPlace.jpg and /dev/null differ
diff --git a/src/public/src/assets/images/websites/earjobs.jpg b/src/public/src/assets/images/websites/earjobs.jpg
deleted file mode 100644
index 2df533c..0000000
Binary files a/src/public/src/assets/images/websites/earjobs.jpg and /dev/null differ
diff --git a/src/public/src/assets/images/websites/kopalife.jpg b/src/public/src/assets/images/websites/kopalife.jpg
deleted file mode 100644
index 55377de..0000000
Binary files a/src/public/src/assets/images/websites/kopalife.jpg and /dev/null differ
diff --git a/src/public/src/assets/images/websites/solinvictus.jpg b/src/public/src/assets/images/websites/solinvictus.jpg
deleted file mode 100644
index 76388d6..0000000
Binary files a/src/public/src/assets/images/websites/solinvictus.jpg and /dev/null differ
diff --git a/src/public/src/assets/images/websites/stateofescape.jpg b/src/public/src/assets/images/websites/stateofescape.jpg
deleted file mode 100644
index 5fdb7bc..0000000
Binary files a/src/public/src/assets/images/websites/stateofescape.jpg and /dev/null differ
diff --git a/src/public/src/components/effects/Observe.tsx b/src/public/src/components/effects/Observe.tsx
deleted file mode 100644
index 7532649..0000000
--- a/src/public/src/components/effects/Observe.tsx
+++ /dev/null
@@ -1,29 +0,0 @@
-
-//Intersection Observer
-export type ObserveCallback = (e:IntersectionObserverEntry)=>void;
-export type Observation = {element:HTMLElement,callback:ObserveCallback};
-export const observations:Observation[] = [];
-
-export const observer = typeof IntersectionObserver === "function" ? (
- new IntersectionObserver((entries, observer) => {
- [...observations].forEach(ob => {
- entries.forEach(e => {
- if(ob.element != e.target) return;
- ob.callback(e);
- });
- });
- },{})
-) : null;
-
-export const addObserve = (ob:Observation) => {
- if(!observer) return;
- observations.push(ob);
- observer.observe(ob.element);//TODO: "Is observing"
-}
-
-export const removeObserve = (ob:Observation) => {
- if(!observer) return;
- let i = observations.indexOf(ob);
- if(i !== -1) observations.splice(i, 1);
- observer.unobserve(ob.element);
-}
\ No newline at end of file
diff --git a/src/public/src/components/effects/ScrollFade.tsx b/src/public/src/components/effects/ScrollFade.tsx
deleted file mode 100644
index 781a75f..0000000
--- a/src/public/src/components/effects/ScrollFade.tsx
+++ /dev/null
@@ -1,74 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Durations, Easings } from '@settings/all';
-import { Observation, addObserve, observer, removeObserve } from './Observe';
-
-export type ScrollFadeDirection = 'bottom' | 'top' | 'left' | 'right';
-export type ScrollFadeDelay = 'short' | 'medium' | 'long';
-
-export interface ScrollFadeProps {
- from?:ScrollFadeDirection;
- delay?:ScrollFadeDelay;
- amount?:number;
- visible?:boolean;
- children?:React.ReactNode;
-}
-
-export interface ScrollFadeState {
- visible:boolean; canFade:boolean;
-}
-
-const ScrollFadeWrapper = styled.div(({ visible, ...props }) => {
- const { delay, from, amount } = { from: 'bottom', delay: 'long', amount: 4, ...props };
- const time = delay == 'long' ? Durations.timeLong : delay == 'medium' ? Durations.timeMedium : Durations.timeShort;
-
- return `
- display: inline-block;
- opacity: 0;
- transition:
- opacity ${time}s ${Easings.easeOut},
- transform ${time}s ${Easings.easeOut},
- -webkit-transform ${time}s ${Easings.easeOut}
- ;
- transform: translate(
- ${from == 'left' ? -amount : from == 'right' ? amount : 0}rem,
- ${from == 'top' ? -amount : from == 'bottom' ? amount : 0}rem
- );
-
- ${visible?`
- opacity: 1;
- transform: translate(0,0);
- `:''}
- `
-});
-
-
-export class ScrollFade extends React.Component {
- element:HTMLDivElement | null = null;
- observation:Observation | null = null;
-
- constructor(props:ScrollFadeProps) {
- super(props);
- this.state = { visible: false, canFade: !!observer };
- }
-
- componentDidMount() {
- if(!this.element) return;
- addObserve(this.observation = { element: this.element, callback: e => {
- this.setState({ visible: e.isIntersecting });
- if(this.state.visible && this.observation) removeObserve(this.observation);
- }});
- }
-
- componentWillUnmount() {
- if(this.observation) removeObserve(this.observation);
- }
-
- render() {
- return this.element = e}>
-
- { this.props.children }
-
-
- }
-}
\ No newline at end of file
diff --git a/src/public/src/components/forms/ContactForm.tsx b/src/public/src/components/forms/ContactForm.tsx
deleted file mode 100644
index 872bbec..0000000
--- a/src/public/src/components/forms/ContactForm.tsx
+++ /dev/null
@@ -1,80 +0,0 @@
-import * as React from 'react';
-import { useForm } from 'react-hook-form';
-import { Input, TextArea } from '@objects/interactive/Input';
-import { Button, ButtonGroup } from '@objects/interactive/Button';
-import * as yup from 'yup';
-import { Panel } from '@objects/feedback/Panel';
-import { Heading2 } from '@objects/typography/Heading';
-import { sendMail, SendMailParams } from '@api/SendMail';
-
-export interface ContactFormProps {
-
-}
-
-const FormValidator = yup.object().shape({
- name: yup.string().max(128).required(),
- email: yup.string().email().required(),
- message: yup.string().required()
-})
-
-export const ContactForm = (props:ContactFormProps) => {
- const { register, handleSubmit, errors, formState } = useForm({
- validationSchema: FormValidator, mode: 'onChange'
- });
-
- const [ pending, setPending ] = React.useState(false);
- const [ success, setSuccess ] = React.useState(false);
-
- const onSubmit = async (data:SendMailParams) => {
- console.log(data);
- setPending(true);
-
- await sendMail(data);
-
- setPending(false);
- setSuccess(true);
- console.log('Send', pending);
- };
-
- return success ? <>
-
- Thanks for contacting
-
-
- Thank you for your message, I will be in touch shortly to follow up
- (generally within a couple of days).
-
-
- In the meantime why not check out my social channels?
-
- > : (
-
- )
-}
\ No newline at end of file
diff --git a/src/public/src/components/layout/Background.tsx b/src/public/src/components/layout/Background.tsx
deleted file mode 100644
index 3a00b0b..0000000
--- a/src/public/src/components/layout/Background.tsx
+++ /dev/null
@@ -1,68 +0,0 @@
-import * as React from 'react';
-import styled, { keyframes, css } from 'styled-components';
-import { Easings, Durations } from '@settings/all';
-
-const BackgroundWrapper = styled.div`
- width: 100%;
- height: 100%;
- position: fixed;
- left: 0;
- top: 0;
- z-index: -1;
-`;
-
-const BackgroundImage = styled.svg`
- display: block;
- object-fit: cover;
- min-width: 100%;
- min-height: 100%;
-`;
-
-const BackgroundGradient = styled.stop<{color:string}>(props => `
- stop-color:${props.color};
- stop-opacity:1;
-`);
-
-const BackgroundFadeIn = keyframes`
- from {
- opacity: 0;
- transform: translateY(12%);
- }
-
- to {
- opacity: 0.4;
- transform: translateY(0%);
- }
-`;
-
-const BackgroundCircle = styled( (props:{
- layer:number, fill:string, className?:string
-}) => )(props => css`
- opacity: 0;
- animation: ${BackgroundFadeIn} ${Durations.timeVeryLong}s forwards ${Easings.easeOut};
- animation-delay: ${Durations.timeShort + (props.layer * Durations.timeShort)}s;
-`);
-
-
-export const Background = () => (
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-)
\ No newline at end of file
diff --git a/src/public/src/components/layout/Boundary.tsx b/src/public/src/components/layout/Boundary.tsx
deleted file mode 100644
index ee66a84..0000000
--- a/src/public/src/components/layout/Boundary.tsx
+++ /dev/null
@@ -1,20 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Sizes } from '@settings/all';
-
-export type BoundarySize = 'small' | 'medium' | 'large';
-
-export interface BoundaryProps {
- size?:BoundarySize;
-}
-
-export const Boundary = styled.div(props => `
- margin-left: auto;
- margin-right: auto;
-
- max-width: ${
- props.size === 'large' ? Sizes.laptopLarge :
- props.size === 'small' ? Sizes.mobileLarge :
- Sizes.tabletLarge
- }px;
-`);
\ No newline at end of file
diff --git a/src/public/src/components/layout/Frame.tsx b/src/public/src/components/layout/Frame.tsx
deleted file mode 100644
index 8990a58..0000000
--- a/src/public/src/components/layout/Frame.tsx
+++ /dev/null
@@ -1,31 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Gutters, MediaQueries } from '@settings/all';
-
-export const Frame = styled((p:{
- padded?:boolean;
- children:React.ReactNode;
- className?:string;
-}) => {
- let { padded, ...props } = p;
- if(padded) return
- return <>{props.children}>
-})(props => `
- ${props.padded ? `padding: ${Gutters.extraSmall};`:''}
-
- ${MediaQueries.mobileUp} {
- ${props.padded ? `padding: ${Gutters.small};`:''}
- }
-
- ${MediaQueries.tabletUp} {
- ${props.padded ? `padding: ${Gutters.small} ${Gutters.medium};`:''}
- }
-
- ${MediaQueries.laptopUp} {
- ${props.padded ? `padding: ${Gutters.small} ${Gutters.large};`:''}
- }
-
- ${MediaQueries.desktopUp} {
- ${props.padded ? `padding: ${Gutters.medium} ${Gutters.extraLarge};`:''}
- }
-`);
\ No newline at end of file
diff --git a/src/public/src/components/layout/Section.tsx b/src/public/src/components/layout/Section.tsx
deleted file mode 100644
index 1b54ba6..0000000
--- a/src/public/src/components/layout/Section.tsx
+++ /dev/null
@@ -1,15 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Gutters, MediaQueries } from '@settings/all';
-
-export const Section = styled.section(props => `
- + ${() => Section} {
- margin-top: ${Gutters.extraExtraLarge};
- }
-
- ${MediaQueries.laptopUp} {
- + ${() => Section} {
- margin-top: ${Gutters.extremeLarge};
- }
- }
-`);
\ No newline at end of file
diff --git a/src/public/src/components/navigation/Footer.tsx b/src/public/src/components/navigation/Footer.tsx
deleted file mode 100644
index 8f38024..0000000
--- a/src/public/src/components/navigation/Footer.tsx
+++ /dev/null
@@ -1,101 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Boundary } from '../layout/Boundary';
-import { Logo } from '../../objects/branding/Logo';
-import { Frame } from '../layout/Frame';
-import { Gutters, Colors, MediaQueries, Borders } from '@settings/all';
-import { Link } from 'gatsby';
-
-const FooterInner = styled(Boundary)`
- display: flex;
- align-items: center;
- flex-flow: column;
- padding: ${Gutters.large};
- margin-top: ${Gutters.extremeLarge};
-
- ${MediaQueries.tabletUp} {
- flex-flow: row;
- flex-wrap: wrap;
- justify-content: space-between;
- }
-`;
-
-const FooterLogo = styled(Logo)`
- max-width: 200px;
- margin: ${Gutters.medium} 0;
-`
-
-const FooterUpper = styled(Frame)`
- min-height: 5rem;
- margin-top: ${Gutters.large};
-`;
-
-const FooterCopyright = styled.div`
- margin: 0;
- width: 100%;
- background: ${Colors.primary};
- text-align: center;
- font-size: 0.9em;
- padding: ${Gutters.medium};
-
- ${MediaQueries.tabletUp} {
- text-align: right;
- }
-`
-
-const FooterNav = styled.nav`
- display: flex;
- flex-wrap: wrap;
- justify-content: space-between;
- padding-top: ${Gutters.medium};
- width: 100%;
-
- ${MediaQueries.mobileUp} {
- justify-content: flex-start;
- }
-
- ${MediaQueries.tabletUp} {
- max-width: 75%;
- }
-`;
-
-const FooterLink = styled(Link)`
- color: inherit;
- text-align: center;
- padding: ${Gutters.small} ${Gutters.extraSmall};
- width: calc(50% - ${Gutters.extraSmall});
-
- &:hover { text-decoration: underline; }
-
- ${MediaQueries.mobileUp} {
- width: 25%;
- text-align: left;
- }
-`;
-
-const FooterLine = styled.div`
- width: 100%;
-`;
-
-export const Footer = () => (
-
-
-
-
-
-
-
-
- Contact
- Privacy Policy
-
-
-
-
-
-
- © 2012 ~ {new Date().getFullYear()} - Dominic Masters
-
-
-
-);
\ No newline at end of file
diff --git a/src/public/src/components/navigation/Header.tsx b/src/public/src/components/navigation/Header.tsx
deleted file mode 100644
index a2beb73..0000000
--- a/src/public/src/components/navigation/Header.tsx
+++ /dev/null
@@ -1,33 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { ZIndex, Gutters, MediaQueries, Colors } from '@settings/all';
-import { Boundary } from '../layout/Boundary';
-import { Logo } from '../../objects/branding/Logo';
-
-const HeaderWrapper = styled.header`
- position: fixed;
- z-index: ${ZIndex.header};
- width: 100%;
- pointer-events: none;
- top: 0;
-`;
-
-const HeaderInner = styled(Boundary)`
- display: flex;
- justify-content: space-between;
- align-items: center;
- padding: ${Gutters.medium};
-`;
-
-const HeaderLogo = styled(Logo)`
- max-width: 150px;
- pointer-events: initial;
-`
-
-export const Header = () => (
-
-
-
-
-
-)
\ No newline at end of file
diff --git a/src/public/src/components/page/PageWrapper.tsx b/src/public/src/components/page/PageWrapper.tsx
deleted file mode 100644
index c786da3..0000000
--- a/src/public/src/components/page/PageWrapper.tsx
+++ /dev/null
@@ -1,23 +0,0 @@
-import * as React from 'react';
-import Helmet from 'react-helmet';
-
-export interface PageWrapperProps {
- children:React.ReactNode;
- title:string|null;
-}
-
-export const PageWrapper = ({ title, children }:PageWrapperProps) => {
- if(!title) {
- title = 'domsPlace | Dominic Masters\' Personal Website';
- } else {
- title = `${title} | domsPlace`
- }
-
- return <>
-
-
- { title }
-
- {children}
- >;
-}
\ No newline at end of file
diff --git a/src/public/src/components/sections/BannerImage.tsx b/src/public/src/components/sections/BannerImage.tsx
deleted file mode 100644
index 4576221..0000000
--- a/src/public/src/components/sections/BannerImage.tsx
+++ /dev/null
@@ -1,87 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Section } from '@components/layout/Section';
-import { Frame } from '@components/layout/Frame';
-import { ScrollFade } from '@components/effects/ScrollFade';
-import { Boundary, BoundaryProps } from '@components/layout/Boundary';
-import { MediaQueries } from '@settings/all';
-
-const BannerImageWrapper = styled(Section)`
- position: relative;
- overflow: hidden;
-
- ${MediaQueries.mobileUp} {
- min-height: 25em;
- }
-
- ${MediaQueries.tabletUp} {
- display: flex;
- align-items: center;
- justify-content: flex-end;
- }
-
- ${MediaQueries.laptopUp} {
- min-height: 35em;
- }
-`;
-
-const BannerImageMedia = styled(ScrollFade)`
- position: relative;
- width: 100%;
-
- ${MediaQueries.tabletUp} {
- position: absolute;
- top: 0;
- left: 0;
- width: auto;
- height: 100%;
- }
-`;
-
-const BannerImageContent = styled(ScrollFade)`
- position: relative;
- text-align: center;
-
- ${MediaQueries.tabletUp} {
- width: 80%;
- text-align: left;
- }
-
- ${MediaQueries.laptopUp} {
- width: 70%;
- }
-`;
-
-const BannerImageText = styled.div`
- ${MediaQueries.tabletUp} {
- width: ${90}%;
- transform: translateX(10%);
- }
-`;
-
-export type BannerImageSectionProps = BoundaryProps & {
- className?:string;
- title?:React.ComponentFactory;
- subtitle?:React.ComponentFactory;
- body?:React.ComponentFactory;
-};
-
-export const BannerImageSection = (p:BannerImageSectionProps) => {
- const { title, subtitle, body, ...props } = p;
-
- return
-
-
-
-
-
- { title ? title() : null }
- { subtitle ? subtitle() : null }
-
-
- { body ? body() : null }
-
-
-
-
-};
\ No newline at end of file
diff --git a/src/public/src/components/sections/ButtonTitle.tsx b/src/public/src/components/sections/ButtonTitle.tsx
deleted file mode 100644
index 288168b..0000000
--- a/src/public/src/components/sections/ButtonTitle.tsx
+++ /dev/null
@@ -1,34 +0,0 @@
-import * as React from 'react';
-import { Boundary, BoundaryProps } from '@components/layout/Boundary';
-import { Section } from '@components/layout/Section';
-import styled from 'styled-components';
-import { ScrollFade } from '@components/effects/ScrollFade';
-import { Frame } from '@components/layout/Frame';
-import { Gutters } from '@settings/all';
-
-const ButtonTitleInner = styled(ScrollFade)`
- width: 100%;
-`;
-
-const ButtonTitleContent = styled(Frame)`
- text-align: center;
-`;
-
-export type ButtonTitleProps = BoundaryProps & {
- title?:React.ComponentFactory;
- body?:React.ComponentFactory;
- buttons?:React.ComponentFactory;
-}
-
-export const ButtonTitle = ({ title, body, buttons, ...props }:ButtonTitleProps) => (
-
-
-
- { title ? title() : null }
- { body ? body() : null }
-
- { buttons ? buttons() : buttons }
-
-
-
-);
\ No newline at end of file
diff --git a/src/public/src/components/sections/IconGrid.tsx b/src/public/src/components/sections/IconGrid.tsx
deleted file mode 100644
index 02b3fe0..0000000
--- a/src/public/src/components/sections/IconGrid.tsx
+++ /dev/null
@@ -1,118 +0,0 @@
-import * as React from 'react';
-import styled, { css } from 'styled-components';
-import { Boundary, BoundaryProps } from '@components/layout/Boundary';
-import { Section } from '@components/layout/Section';
-import { Frame } from '@components/layout/Frame';
-import { Gutters, MediaQueries } from '@settings/all';
-import { ScrollFade } from '@components/effects/ScrollFade';
-
-const IconGridIconSplit = (index:number, spacing:number, rows:number, columns:number) => {
- let rowsWidth = spacing * (rows - 1);
- let halfRowsWidth = (rowsWidth / 2) - rowsWidth;
- let x = halfRowsWidth + (spacing * (index/columns));
-
- return `
- width: ${100 / columns}%;
- transform: translateX(${x}%);
- `;
-}
-
-const IconGridIconWrapper = styled.div((props:{index:number}) => `
- width: 20%;
- padding: ${Gutters.extraSmall};
-
- ${MediaQueries.mobileUp} {
- padding: ${Gutters.small};
- }
-
- ${MediaQueries.tabletUp} {
- ${IconGridIconSplit(props.index ? props.index : 0, 30, 5, 5)}
- }
-
- ${MediaQueries.laptopUp} {
- ${IconGridIconSplit(props.index ? props.index : 0, 50, 5, 5)}
- }
-`);
-
-const IconGridIconInner = styled.div`
- position: relative;
- padding-bottom: 100%;
-`;
-
-const IconGridIconImage = styled.img`
- display: block;
- position: absolute;
- width: 100%;
- height: 100%;
- top: 0;
- left: 0;
- object-fit: contain;
- object-position: center;
-`;
-
-export type IconGridIconProps = {
- index?:number;
- title:string;
- image:any;
-}
-
-const IconGridIcon = (props:IconGridIconProps) => (
-
-
-
-
-
-);
-
-
-const IconGridWrapper = styled((p:BoundaryProps) => )`
- display: flex;
- flex-direction: column;
- align-items: center;
-`;
-
-const IconGridContent = styled(Frame)`
- width: 100%;
- display: inline-block;
- text-align: center;
- margin-bottom: ${Gutters.extraSmall};
-
- ${MediaQueries.mobileUp} {
- width: auto;
- margin-bottom: ${Gutters.small};
- }
-`;
-
-const IconGridGrid = styled(ScrollFade)`
- width: 100%;
-`;
-
-const IconGridInner = styled.div`
- display: flex;
- flex-wrap: wrap;
- margin: 0 auto;
-
- ${MediaQueries.tabletUp} {
- max-width: 700px;
- }
-`;
-
-
-export type IconGridProps = BoundaryProps & {
- title?:React.ComponentFactory;
- icons?:IconGridIconProps[];
-}
-
-export const IconGrid = ({ icons, title, ...props }:IconGridProps) => (
-
-
- { title ? title() : null }
-
-
-
-
- { icons ? icons.map((icon,i) => ) : null }
-
-
-
-);
\ No newline at end of file
diff --git a/src/public/src/components/sections/Mosaic.tsx b/src/public/src/components/sections/Mosaic.tsx
deleted file mode 100644
index 599a395..0000000
--- a/src/public/src/components/sections/Mosaic.tsx
+++ /dev/null
@@ -1,117 +0,0 @@
-import * as React from 'react';
-import { ScrollFade, ScrollFadeProps } from '@components/effects/ScrollFade';
-import { Section } from '@components/layout/Section';
-import { Frame } from '@components/layout/Frame';
-import styled from 'styled-components';
-import { Boundary, BoundaryProps } from '@components/layout/Boundary';
-import { MediaQueries, Durations, Easings } from '@settings/all';
-import Img, { GatsbyImageProps, FluidObject } from 'gatsby-image';
-
-const MosaicWrapper = styled(Section)`
- ${MediaQueries.tabletUp} {
- display: flex;
- align-items: center;
- }
-`;
-
-const MosaicGrid = styled.div`
- position: relative;
- width: 100%;
-
- ${MediaQueries.tabletUp} {
- width: 50%;
- order: 2;
- }
-`;
-
-const MosaicPad = styled.div`
- width: 100%;
- padding-bottom: 80%;
-`;
-
-const MosaicContent = styled(ScrollFade)`
- text-align: center;
- ${MediaQueries.tabletUp} {
- width: 50%;
- text-align: left;
- }
-`;
-
-const MosaicBody = styled.div`
- ${MediaQueries.tabletUp} {
- width: 90%;
- transform: translateX(10%);
- }
-`;
-
-export type MosaicTileProps = ScrollFadeProps & {
- index?:number;
- image:GatsbyImageProps;
- to:string;
- className?:string;
-}
-
-const MosaicTileImage = styled((props:GatsbyImageProps) => )`
- width: 100%;
- display: block;
-`
-
-const MosaicTile = styled( ({ image, to, ...props }:MosaicTileProps) => {
- return
-
-
-
- ;
-})(props => `
- position: absolute;
- transition: all ${Durations.timeShort}s ${Easings.easeOut};
-
- ${props.index == 0 ? `
- bottom: 10%;
- left: 15%;
- width: 50%;
- z-index: 3;
-
- &:hover { transform: translateY(-0.5em); }
- ` : props.index == 1 ? `
- bottom: 20%;
- left: 25%;
- width: 45%;
-
- &:hover { transform: translateX(-0.5em); }
- ` : props.index == 2 ? `
- width: 50%;
- right: 5%;
- top: 25%;
-
- &:hover { transform: translate(0.5em, -0.5em); }
- ` : ''}
-`);
-
-export type MosaicProps = BoundaryProps & {
- title:React.ComponentFactory;
- body?:React.ComponentFactory;
- images?:MosaicTileProps[];
-}
-
-export const Mosaic = styled((p:MosaicProps) => {
- let { title, body, images, ...props } = p;
- return
-
- { images ? images.map((c,i) => ) : null }
-
-
-
-
-
- {title ? title() : null}
- {body ? body() : null}
-
-
- ;
-})`
- ${MediaQueries.tabletUp} {
- display: flex;
- align-items: center;
- }
-`;
\ No newline at end of file
diff --git a/src/public/src/components/sections/SplitFrames.tsx b/src/public/src/components/sections/SplitFrames.tsx
deleted file mode 100644
index e0a5d36..0000000
--- a/src/public/src/components/sections/SplitFrames.tsx
+++ /dev/null
@@ -1,98 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Section } from '@components/layout/Section';
-import { MediaQueries, Gutters } from '@settings/all';
-import { BoundaryProps, Boundary } from '@components/layout/Boundary';
-import { ScrollFade, ScrollFadeProps } from '@components/effects/ScrollFade';
-import { Frame } from '@components/layout/Frame';
-
-const SplitFramesContainer = styled(Section)`
- ${MediaQueries.tabletUp} {
- display: flex;
- align-items: center;
- flex-wrap: wrap;
- }
-
- ${MediaQueries.laptopUp} {
- justify-content: space-between;
- }
-`;
-
-const SplitFramesHeader = styled(ScrollFade)`
- width: 100%;
- text-align: center;
-
- ${MediaQueries.tabletUp} {
- margin-bottom: ${Gutters.large};
- }
-`;
-
-type SplitProps = ScrollFadeProps & { padded?:boolean };
-const Split = styled((props:SplitProps) => (
-
-
- { props.children }
-
-
-))`
- width: 100%;
- text-align: center;
-
- ${MediaQueries.tabletUp} {
- width: 50%;
- margin: 0;
- text-align: left;
- }
-
- ${MediaQueries.laptopUp} {
- width: calc(50% - (${Gutters.medium}/2));
- }
-
- ${MediaQueries.desktopUp} {
- width: calc(50% - (${Gutters.large}/2));
- }
-`;
-
-const RightSplit = Split;
-const LeftSplit = styled(Split)`
- width: 100%;
- margin-bototm: ${Gutters.extraLarge};
-`;
-
-
-
-export type SplitFramesProps = BoundaryProps & {
- title?:React.ComponentFactory;
- subtitle?:React.ComponentFactory;
- leftOptions?:SplitProps;
- rightOptions?:SplitProps;
- left:React.ComponentFactory;
- right:React.ComponentFactory;
-};
-
-export const SplitFrames = (props:SplitFramesProps) => {
- let header;
-
- if(props.title || props.subtitle) {
- header =
-
- { props.title ? props.title() : null }
- { props.subtitle ? props.subtitle() : null }
-
-
- }
-
- return (
-
- { header }
-
-
- {props.left()}
-
-
-
- {props.right()}
-
-
- );
-};
\ No newline at end of file
diff --git a/src/public/src/components/sections/StackedMosaic.tsx b/src/public/src/components/sections/StackedMosaic.tsx
deleted file mode 100644
index 9da4b6c..0000000
--- a/src/public/src/components/sections/StackedMosaic.tsx
+++ /dev/null
@@ -1,99 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Boundary, BoundaryProps } from '@components/layout/Boundary';
-import { Section } from '@components/layout/Section';
-import { MediaQueries, Easings, Durations } from '@settings/all';
-import { ScrollFade, ScrollFadeProps } from '@components/effects/ScrollFade';
-import { Frame } from '@components/layout/Frame';
-import Img, { GatsbyImageProps } from 'gatsby-image';
-
-const StackedMosaicWrapper = styled((props:BoundaryProps) => )`
- ${MediaQueries.tabletUp} {
- display: flex;
- align-items: center;
- }
-`;
-
-const StackedMosaicGrid = styled.div`
- position: relative;
- overflow: hidden;
- padding-bottom: 72%;
-
- ${MediaQueries.tabletUp} {
- width: 50%;
- padding-bottom: 33%;
- }
-`;
-
-const StackedMosaicContent = styled(ScrollFade)`
- text-align: center;
- ${MediaQueries.tabletUp} {
- width: 50%;
- text-align: left;
- }
-`;
-
-const StackedMosaicHeader = styled(Frame)`
- ${MediaQueries.desktopUp} {
- position: relative;
- transform: translateX(-2em);
- }
-`
-
-const StackedMosaicBody = styled(Frame)`
- ${MediaQueries.desktopUp} {
- position: relative;
- transform: translateX(3em);
- }
-`;
-
-type StackedMosaicTileProps = ScrollFadeProps & {
- index?:number;
- to:string;
- image:GatsbyImageProps;
-}
-
-const StackedMosaicTile = styled( ({ to, image, ...props}:StackedMosaicTileProps) => (
-
-
-
-))(props => `
- position: absolute;
- width: 50%;
- margin: 0;
- transition: all ${Durations.timeShort}s ${Easings.easeOut};
- transform: translateY(0em) rotate(-25deg);
-
- ${props.index ? `
- left: ${(10 * props.index) + 2.5}%;
- top: ${14 + (props.index * 2)}%;
- ` : ``}
-
- &:hover {
- transform: translateY(-0.5em) rotate(-25deg);
- }
-`);
-
-export type StackedMosaicProps = BoundaryProps & {
- images:StackedMosaicTileProps[];
- title:React.ComponentFactory;
- body:React.ComponentFactory;
-}
-
-export const StackedMosaic = ({ title, body, images, ...p }:StackedMosaicProps) => (
-
-
- {images.map((e,i) => )}
-
-
-
-
-
- { title ? title() : null }
-
-
- { body ? body() : null }
-
-
-
-);
\ No newline at end of file
diff --git a/src/public/src/layouts/main.tsx b/src/public/src/layouts/main.tsx
deleted file mode 100644
index 48ee4b5..0000000
--- a/src/public/src/layouts/main.tsx
+++ /dev/null
@@ -1,65 +0,0 @@
-import * as React from 'react';
-import { Header } from '@components/navigation/Header';
-import { Background } from '../components/layout/Background';
-import { Footer } from '../components/navigation/Footer';
-import styled, { createGlobalStyle } from 'styled-components';
-import { Gutters, Easings, Durations } from '@settings/all';
-import { BodyStyles, AllElementStyles, AnchorStyles } from '@styles/index';
-import { TransitionGroup, Transition } from "react-transition-group"
-import { TransitionStatus } from 'react-transition-group/Transition';
-
-//Load Global Styles
-const GlobalStyles = createGlobalStyle`
- ${AllElementStyles}
- ${BodyStyles}
- ${AnchorStyles}
-`;
-
-//Transition
-const Timeouts = { enter: Durations.timeMedium * 1000, exit: Durations.timeMedium * 1000 };
-const LayoutTransition = styled.div(({ status }:{ status:TransitionStatus }) => {
- return `
- transition: transform ${Timeouts.enter}ms ${Easings.easeOut};
- ${ status == 'entering' ? `
- position: absolute;
- opacity: 0;
- transform: translateY(-${Gutters.extraExtraLarge});
- `: status == 'entered' ? `
- opacity: 1;
- transform: translateY(0em);
- `: status == 'exiting' ? `
- opacity: 0;
- transform: translateY(${Gutters.extraExtraLarge});
- `:''}
- `;
-});
-
-
-//Main area
-const LayoutMain = styled.main`
- margin-top: ${Gutters.extraExtraLarge};
-`;
-
-export interface LayoutProps {
- children:React.ReactNode;
- location:any;
-};
-export const Layout = ({ location, children, ...props }:LayoutProps) => (
-
-
-
-
-
-
-
- {status =>
- { children }
- }
-
-
-
-
-
-);
-
-export default Layout;
\ No newline at end of file
diff --git a/src/public/src/objects/branding/Logo.tsx b/src/public/src/objects/branding/Logo.tsx
deleted file mode 100644
index 4376cbc..0000000
--- a/src/public/src/objects/branding/Logo.tsx
+++ /dev/null
@@ -1,26 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Link } from 'gatsby';
-import { Colors } from '@settings/all';
-
-export type LogoProps = {
- to?:string;
-};
-
-const LogoImageWrapper = styled.svg`
- display: block;
- width: 100%;
- fill: ${Colors.text};
-`;
-
-export const Logo = styled((props:LogoProps) => (
-
-
-
-
-
-))`
- display: block;
-`;
\ No newline at end of file
diff --git a/src/public/src/objects/code/CodeBlock.tsx b/src/public/src/objects/code/CodeBlock.tsx
deleted file mode 100644
index 51bfe5e..0000000
--- a/src/public/src/objects/code/CodeBlock.tsx
+++ /dev/null
@@ -1,12 +0,0 @@
-import * as React from 'react';
-import SyntaxHighlighter, { SyntaxHighlighterProps } from 'react-syntax-highlighter';
-import test from 'react-syntax-highlighter/dist/esm/styles/hljs/tomorrow-night-eighties';
-import styled from 'styled-components';
-
-export type CodeBlockProps = SyntaxHighlighterProps;
-
-export const CodeBlock = styled((props:CodeBlockProps) => {
- return
-})`
- font-size: 12px;
-`
\ No newline at end of file
diff --git a/src/public/src/objects/feedback/Loader.tsx b/src/public/src/objects/feedback/Loader.tsx
deleted file mode 100644
index 99239de..0000000
--- a/src/public/src/objects/feedback/Loader.tsx
+++ /dev/null
@@ -1,54 +0,0 @@
-import * as React from 'react';
-import styled, { keyframes } from 'styled-components';
-import { Durations, Easings, Colors } from '@settings/all';
-
-const LoaderSpin = keyframes`
- 0% { transform: rotate(0deg); }
- 100% { transform: rotate(360deg); }
-`;
-
-export const LoaderSpinner = styled(props => (
-
-
-
-
-
-
-
-
-
-))`
- display: block;
- width: 1.5em;
- height: 1.5em;
- max-width: 100%;
- max-height: 100%;
-
- animation: ${LoaderSpin} ${Durations.timeLong}s infinite ${Easings.easeInOut};
-
- * {
- stroke: ${Colors.text};
- }
-`
-
-const LoaderInner = styled.div`
- position: absolute;
- left: 50%;
- top: 50%;
- transform: translate(-50%, -50%);
-`
-
-export const Loader = styled(props => (
-
-
-
-
-
-))`
- position: absolute;
- width: 100%;
- height: 100%;
- left: 0;
- top: 0;
- background: rgba(0,0,0,0.5);
-`;
\ No newline at end of file
diff --git a/src/public/src/objects/feedback/Panel.tsx b/src/public/src/objects/feedback/Panel.tsx
deleted file mode 100644
index 8df8ba8..0000000
--- a/src/public/src/objects/feedback/Panel.tsx
+++ /dev/null
@@ -1,18 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { WidgetProps, WidgetStyles } from '@settings/all';
-
-export type PanelProps = WidgetProps & {
-
-}
-
-export const Panel = styled.div`
- ${WidgetStyles};
- display: block;
-`;
-
-export const ErrorPanel = (props:{error:{message:string}}) => {
- let s = props.error.message;
- s = s.charAt(0).toUpperCase() + s.slice(1)
- return {s}
-}
\ No newline at end of file
diff --git a/src/public/src/objects/interactive/Button.tsx b/src/public/src/objects/interactive/Button.tsx
deleted file mode 100644
index 21d2a0d..0000000
--- a/src/public/src/objects/interactive/Button.tsx
+++ /dev/null
@@ -1,43 +0,0 @@
-import * as React from 'react';
-import styled from 'styled-components';
-import { Link } from 'gatsby';
-import { Colors, Gutters, Borders, WidgetStyles, WidgetProps } from '@settings/all';
-import { Loader } from '@objects/feedback/Loader';
-
-export type ButtonType = 'button' | 'reset' | 'submit';
-
-export type ButtonProps = WidgetProps & {
- type?:ButtonType;
- to?:string;
- href?:string;
- children?:React.ReactChild;
- value?:string;
- pending?:boolean;
-}
-
-const StyledButton = styled.button(props => `
- ${WidgetStyles(props, { hover: true })};
- cursor: pointer;
-`);
-
-export const Button = ({ pending, children, ...props }:ButtonProps) => {
- let type = undefined;
-
- if(props.to) {
- type = Link;
- } else if(props.href) {
- type = 'a';
- } else if(props.type) {
- }
-
- return
- { children }
- { pending ? : null }
-
-}
-
-export const ButtonGroup = styled.div`
- ${StyledButton} + ${StyledButton} {
- margin-left: ${Gutters.small};
- }
-`;
\ No newline at end of file
diff --git a/src/public/src/objects/interactive/Input.tsx b/src/public/src/objects/interactive/Input.tsx
deleted file mode 100644
index 2cf7939..0000000
--- a/src/public/src/objects/interactive/Input.tsx
+++ /dev/null
@@ -1,87 +0,0 @@
-import * as React from 'react';
-import styled, { css } from 'styled-components';
-import { StylePlaceholder } from '@tools/styles';
-import { Durations, Easings, Gutters, WidgetStyles, Colors, WidgetProps } from '@settings/all';
-import { Button, ButtonGroup } from './Button';
-import { Panel, ErrorPanel } from '@objects/feedback/Panel';
-
-const InputStyles = (props:InputProps) => css`
- ${WidgetStyles(props)};
- width: 100%;
- display: block;
- background: none;
-
- ${StylePlaceholder(`
- color: inherit;
- font-family: inherit;
- font-weight: inherit;
- opacity: 0.2;
- transition: all ${Durations.timeShort}s ${Easings.easeOut};
- `)}
-`;
-
-const InputError = (props:{ children?:React.Children }) => (
-
-);
-
-const InputInput = styled.input(InputStyles);
-const TextareaInput = styled(styled.textarea(InputStyles))`
- min-width: 100%;
- max-width: 100%;
- min-height: 3.5em;
- resize: vertical;
-`;
-
-const InputLabel = styled.label`
- position: absolute;
- display: block;
- font-size: 0.9em;
- top: 0;
- left: ${Gutters.small};
- line-height: 1;
- transform: translateY(-50%);
- padding: ${Gutters.extraSmall} ${Gutters.small};
- z-index: 2;
- background: ${Colors.background};
- color: ${Colors.text};
-`;
-
-const InputWrapper = styled.div(props => `
- display: block;
- width: 100%;
- position: relative;
-`);
-
-
-export type InputType = (
- //HTML5:
- "color" | "date" | "datetime-local" | "email" | "month" | "number" | "range" |
- "search" | "tel" | "time" | "url" | "week" |
-
- //HTML4~
- "text" | "hidden" | "password" | "radio" | "checkbox" | "textarea" | "file" |
- "image"
-);
-
-export type InputProps = WidgetProps & {
- label?:string;
- type?:InputType;
- register?:any;
- error?:any;
-}
-
-export const Input = ({ error, label, ...props }:JSX.IntrinsicElements['input'] & InputProps) => {
- return
- { label ? { label } : null }
-
- { error ? : null }
-
-};
-
-export const TextArea = ({ error, label, ...props }:JSX.IntrinsicElements['textarea'] & InputProps) => {
- return
- { label ? { label } : null }
-
- { error ? : null }
-
-};
\ No newline at end of file
diff --git a/src/public/src/objects/media/Image.tsx b/src/public/src/objects/media/Image.tsx
deleted file mode 100644
index 873355d..0000000
--- a/src/public/src/objects/media/Image.tsx
+++ /dev/null
@@ -1,16 +0,0 @@
-import { graphql } from "gatsby";
-import { FluidObject } from 'gatsby-image';
-
-export const fluidImage = graphql`
- fragment fluidImage on File {
- childImageSharp {
- fluid(maxWidth: 1000) {
- ...GatsbyImageSharpFluid
- }
- }
- }
-`;
-
-export interface FluidImage {
- childImageSharp:FluidObject;
-}
\ No newline at end of file
diff --git a/src/public/src/objects/typography/Heading.tsx b/src/public/src/objects/typography/Heading.tsx
deleted file mode 100644
index 0490d03..0000000
--- a/src/public/src/objects/typography/Heading.tsx
+++ /dev/null
@@ -1,40 +0,0 @@
-import styled, { css } from "styled-components";
-import { Fonts, FontWeights, FontSizes } from "@settings/all";
-
-
-const Heading = css`
- font-family: ${Fonts.heading};
- display: block;
- font-weight: ${FontWeights.heading};
- margin: 0.4em 0 0.5em;
-`;
-
-export const Heading1 = styled.h1`
- ${Heading}
- font-size: ${FontSizes.heading1};
-`;
-
-export const Heading2 = styled.h2`
- ${Heading}
- font-size: ${FontSizes.heading2}
-`;
-
-export const Heading3 = styled.h3`
- ${Heading}
- font-size: ${FontSizes.heading3};
-`;
-
-export const Heading4 = styled.h4`
- ${Heading}
- font-size: ${FontSizes.heading4};
-`;
-
-export const Heading5 = styled.h5`
- ${Heading}
- font-size: ${FontSizes.heading5};
-`;
-
-export const Heading6 = styled.h6`
- ${Heading}
- font-size: ${FontSizes.heading6};
-`;
\ No newline at end of file
diff --git a/src/public/src/objects/typography/Title.tsx b/src/public/src/objects/typography/Title.tsx
deleted file mode 100644
index 2c92555..0000000
--- a/src/public/src/objects/typography/Title.tsx
+++ /dev/null
@@ -1,23 +0,0 @@
-import * as React from 'react';
-import styled, { css } from 'styled-components';
-import { FontSizes, Fonts, FontWeights, MediaQueries } from '@settings/all';
-import { Heading2, Heading1 } from './Heading';
-
-export const Subtitle = styled(Heading2)`
-`;
-
-export const Title = styled(Heading1)((props:{large?:boolean}) => `
- font-size: ${FontSizes.title};
- margin: 0.4em 0 0.5em;
-
- + ${Subtitle} {
- margin-top: -1em;
- margin-bottom: 1em;
- }
-
- ${props.large ? `
- ${MediaQueries.tabletUp} {
- font-size: ${FontSizes.titleLarge};
- }
- `:''}
-`);
\ No newline at end of file
diff --git a/src/public/src/pages/404.tsx b/src/public/src/pages/404.tsx
deleted file mode 100644
index 71e3da5..0000000
--- a/src/public/src/pages/404.tsx
+++ /dev/null
@@ -1,53 +0,0 @@
-import * as React from 'react';
-import { PageWrapper } from '@components/page/PageWrapper';
-import { Title, Subtitle } from '@objects/typography/Title';
-import { SplitFrames } from '@components/sections/SplitFrames';
-import { CodeBlock } from '@objects/code/CodeBlock';
-import { Heading3 } from '@objects/typography/Heading';
-import { Button } from '@objects/interactive/Button';
-import { Link } from 'gatsby';
-
-const FAKE_CODE = `import { usePageData } from '@data/page';
-
-const thisPage = usePageData(data => {
- if(!data) return
- return
-});
-
-const MyCool404Page = () => <>
- 404 - Not Found
-
- The error... not the internet
- joke about the error.
-
-
-
- \${require(__filename).toString()}
-
-
->;
-`
-
-export default () => {
- return
- 404 - Not Found }
- subtitle={() => The error... Not the joke about the error. }
- left={() =>
- {FAKE_CODE}
- }
- leftOptions={{ padded: true }}
-
- right={() => <>
- Have no fear
-
- Just because we don't have what you're looking for this time, doesn't
- mean we don't offer other cool things.
-
- Continue Browsing
- >}
- rightOptions={{ padded: false }}
- />
-
-}
\ No newline at end of file
diff --git a/src/public/src/pages/contact.tsx b/src/public/src/pages/contact.tsx
deleted file mode 100644
index 2cf488b..0000000
--- a/src/public/src/pages/contact.tsx
+++ /dev/null
@@ -1,37 +0,0 @@
-import * as React from 'react';
-import { PageWrapper } from '@components/page/PageWrapper';
-import { Title } from '@objects/typography/Title';
-import { SplitFrames } from '@components/sections/SplitFrames';
-import { Heading2 } from '@objects/typography/Heading';
-import { ContactForm } from '@components/forms/ContactForm';
-
-export default () => (
-
- Contact Me }
-
- leftOptions={{ padded: true }}
- left={() =>
-
Send a message down the wire
-
- Feel free to reach out, I usually respond within a few days.
-
-
-
- If you prefer to call, then leave your phone number and what times
- you're available and I'll get in touch!
-
-
-
- Looking for something a bit more instant? Get in touch with me via
- social media.
-
-
- {/*
*/}
-
}
-
- rightOptions={{padded: true}}
- right={() => }
- />
-
-);
\ No newline at end of file
diff --git a/src/public/src/pages/index.tsx b/src/public/src/pages/index.tsx
deleted file mode 100644
index dd3fa89..0000000
--- a/src/public/src/pages/index.tsx
+++ /dev/null
@@ -1,175 +0,0 @@
-import * as React from 'react';
-import { PageWrapper } from '@components/page/PageWrapper';
-import { BannerImageSection } from '@components/sections/BannerImage';
-import { Title, Subtitle } from '@objects/typography/Title';
-import { SplitFrames } from '@components/sections/SplitFrames';
-import { Heading1, Heading2 } from '@objects/typography/Heading';
-import { Mosaic } from '@components/sections/Mosaic';
-import { graphql } from 'gatsby';
-import { FluidImage } from '@objects/media/Image';
-import { StackedMosaic } from '@components/sections/StackedMosaic';
-import { IconGrid } from '@components/sections/IconGrid';
-import { ButtonTitle } from '@components/sections/ButtonTitle';
-import { Button } from '@objects/interactive/Button';
-
-export const pageQuery = graphql`
- query {
- bundleImage: file(relativePath: { eq: "websites/bundlfresh.jpg" }) { ...fluidImage }
- cocksoxImage: file(relativePath: { eq: "websites/cocksox.jpg" }) { ...fluidImage }
- soeImage: file(relativePath: { eq: "websites/stateofescape.jpg" }) { ...fluidImage }
-
- kopaImage: file(relativePath: { eq: "websites/kopalife.jpg" }) { ...fluidImage }
- earjobsImage: file(relativePath: { eq: "websites/earjobs.jpg" }) { ...fluidImage }
- solImage: file(relativePath: { eq: "websites/solinvictus.jpg" }) { ...fluidImage }
- }
-`;
-
-export interface IndexPageProps {
- data: {
- bundleImage: FluidImage;
- cocksoxImage: FluidImage;
- soeImage: FluidImage;
- kopaImage: FluidImage;
- earjobsImage: FluidImage;
- solImage: FluidImage;
- };
-}
-
-export default ({ data }:IndexPageProps) => {
- return
- Dominic Masters }
- subtitle={() => Developer, nerd, occasionally funny. }
- body={() =>
- I'm just a nerd with a passion for coding, coffee, and video games.
- Programming since before the internet was cool.
-
}
- />
-
- <>
- Programmer
- I know what I'm doing, sometimes.
- >}
-
- right={() => <>
-
- I am a programmer, born and bred. I have been programming since I
- was around 12 years old and continue to advance my skills more and
- more everyday.
-
-
- Programming is my work and my passion. With over { new Date().getFullYear() - 2007 } years of experience,
- and countless lines of code written, there isn't much I can't develop.
-
- >}
- />
-
- Shopify Plus }
- body={() => <>
-
- I'm currently working full-time as a Senior Full-Stack Developer for
- Shopify Plus at Process Creative .
- I have been working with the platform every day for
- over { new Date().getFullYear() - 2017} years, and enjoy working with
- it immensely.
-
-
-
- Working with Liquid, REST and GraphQL App Development and general
- Shopify tools development, I have had the privilage of working with
- over 40 different Shopify Plus clients, and over 50 Shopify core
- clients.
-
-
-
- Despite Shopify's seemingly limited development environment, I have
- been able to make it do tricks thought impossible. I love finding
- unique solution's to Shopify's limitations, and will continuously
- find ways to surprise everyone, including myself.
-
- >}
- images={[
- { to: '//bundlfresh.com.au', image: data.bundleImage.childImageSharp, delay: 'short' },
- { to: '//cocksox.com', image: data.cocksoxImage.childImageSharp, delay: 'medium' },
- { to: '//www.stateofescape.com', image: data.soeImage.childImageSharp, delay: 'long' },
- ]}
- />
-
-
- Full-Stack Web Dev }
- body={() => <>
-
- I have spent over { new Date().getFullYear() - 2010 } years working
- with both modern and traditional web tech stacks, including NodeJS,
- TypeScript, React, ES6, Webpack, Babel, SCSS, PHP, ASP, SQL and more.
-
-
-
- My specialty is making beautiful and interactive online web experiences.
- Why must web suck? I am to prove that it doesn't always have to.
-
- >
- } />
-
- Platforms I work with }
- icons={[
- /* First Row */
- { title: "C#", image: require('@assets/images/branding/csharp/csharp-logo.svg') },
- { title: "NodeJS", image: require('@assets/images/branding/nodejs/nodejs-logo.svg') },
- { title: "Java", image: require('@assets/images/branding/java/java-logo.svg') },
- { title: "PHP", image: require('@assets/images/branding/php/php-logo.svg') },
- { title: 'C++', image: require('@assets/images/branding/cpp/cpp-logo.svg') },
-
- /* Second Row */
- { title: 'TypeScript', image: require('@assets/images/branding/typescript/typescript-logo.svg') },
- { title: 'React', image: require('@assets/images/branding/react/react-logo.svg') },
- { title: 'Redux', image: require('@assets/images/branding/redux/redux-logo.svg') },
- { title: 'webpack', image: require('@assets/images/branding/webpack/webpack-logo.svg') },
- { title: 'jQuery', image: require('@assets/images/branding/jquery/jquery-logo.svg') },
-
- /* Third Row */
- { title: 'Shopify', image: require('@assets/images/branding/shopify/shopify-logo.svg') },
- { title: 'Heroku', image: require('@assets/images/branding/heroku/heroku-logo.svg' ) },
- { title: 'Google Cloud Platform', image: require('@assets/images/branding/google-cloud/google-cloud-logo.svg') },
- { title: 'Digital Ocean', image: require('@assets/images/branding/digitalocean/digitalocean-logo.svg') },
- { title: 'neto', image: require('@assets/images/branding/neto/neto-logo.svg') },
-
- /* Fourth Row */
- { title: 'MonoGame', image: require('@assets/images/branding/monogame/monogame-logo.svg') },
- { title: 'OpenGL', image: require('@assets/images/branding/opengl/opengl-logo.svg') },
- { title: 'Unity', image: require('@assets/images/branding/unity/unity-logo.svg') },
- { title: 'LWJGL', image: require('@assets/images/branding/lwjgl/lwjgl-logo.svg') },
-
- /* Fifth Row */
- { title: 'GraphQL', image: require('@assets/images/branding/graphql/graphql-logo.svg') },
- { title: 'MySQL', image: require('@assets/images/branding/mysql/mysql-logo.svg') },
- { title: 'PostgreSQL', image: require('@assets/images/branding/pgsql/pgsql-logo.svg') },
-
- /* Sixth Row */
- { title: 'Discord', image: require('@assets/images/branding/discord/discord-logo.svg') },
- { title: 'Twitch', image: require('@assets/images/branding/twitch/twitch-logo.svg') },
- { title: 'Twitter', image: require('@assets/images/branding/twitter/twitter-logo.svg') }
- ]}
- />
-
- Get in touch }
- body={() =>
- Want to get in touch, pick my brain or just have a chat?
- Head over to my contact page and feel free to reach out.
-
}
- buttons={() => Contact Me }
- />
-
-}
\ No newline at end of file
diff --git a/src/public/src/pages/legal/privacy.tsx b/src/public/src/pages/legal/privacy.tsx
deleted file mode 100644
index 7952863..0000000
--- a/src/public/src/pages/legal/privacy.tsx
+++ /dev/null
@@ -1,164 +0,0 @@
-import * as React from 'react';
-import { PageWrapper } from '@components/page/PageWrapper';
-import { Boundary } from '@components/layout/Boundary';
-import { Title } from '@objects/typography/Title';
-import { Heading4, Heading3, Heading2 } from '@objects/typography/Heading';
-
-export default () => (
-
-
- Privacy Policy
- Effective date: June 27, 2018
-
- domsPlace ("us", "we", or "our") operates
- the https://domsplace.com website (the
- "Service").
-
-
-
- This page informs you of our policies regarding the collection, use, and
- disclosure of personal data when you use our Service and the choices you
- have associated with that data.
-
-
-
- We use your data to provide and improve the Service. By using the
- Service, you agree to the collection and use of information in
- accordance with this policy. Unless otherwise defined in this Privacy
- Policy, terms used in this Privacy Policy have the same meanings as in
- our Terms and Conditions, accessible from
- https://domsplace.com
-
-
-
-
- Information Collection And Use
-
- We collect several different types of information for various purposes
- to provide and improve our Service to you.
-
-
- Types of Data Collected
- Personal Data
-
- While using our Service, we may ask you to provide us with certain
- personally identifiable information that can be used to contact or
- identify you ("Personal Data"). Personally identifiable information may
- include, but is not limited to:
-
-
- Email address
- First name and last name
- Phone number
- Cookies and Usage Data
-
-
- Usage Data
-
- We may also collect information how the Service is accessed and used =
- ("Usage Data"). This Usage Data may include information such as your
- computer's Internet Protocol address (e.g. IP address), browser type,
- browser version, the pages of our Service that you visit, the time and
- date of your visit, the time spent on those pages, unique device
- identifiers and other diagnostic data.
-
-
- Tracking & Cookies Data
-
- We use cookies and similar tracking technologies to track the activity
- on our Service and hold certain information.
-
-
- Cookies are files with small amount of data which may include an
- anonymous unique identifier. Cookies are sent to your browser from a
- website and stored on your device. Tracking technologies also used are
- beacons, tags, and scripts to collect and track information and to
- improve and analyze our Service.
-
-
- You can instruct your browser to refuse all cookies or to indicate when
- a cookie is being sent. However, if you do not accept cookies, you may
- not be able to use some portions of our Service.
-
- Examples of Cookies we use:
-
- Session Cookies. We use Session Cookies to operate our Service.
- Preference Cookies. We use Preference Cookies to remember your preferences and various settings.
- Security Cookies. We use Security Cookies for security purposes.
-
-
- Use of Data
- domsPlace uses the collected data for various purposes:
-
- To provide and maintain the Service
- To notify you about changes to our Service
- To allow you to participate in interactive features of our Service when you choose to do so
- To provide customer care and support
- To provide analysis or valuable information so that we can improve the Service
- To monitor the usage of the Service
- To detect, prevent and address technical issues
-
-
- Transfer Of Data
- Your information, including Personal Data, may be transferred to — and maintained on — computers located outside of your state, province, country or other governmental jurisdiction where the data protection laws may differ than those from your jurisdiction.
- If you are located outside Australia and choose to provide information to us, please note that we transfer the data, including Personal Data, to Australia and process it there.
- Your consent to this Privacy Policy followed by your submission of such information represents your agreement to that transfer.
- domsPlace will take all steps reasonably necessary to ensure that your data is treated securely and in accordance with this Privacy Policy and no transfer of your Personal Data will take place to an organization or a country unless there are adequate controls in place including the security of your data and other personal information.
-
- Disclosure Of Data
-
- Legal Requirements
- domsPlace may disclose your Personal Data in the good faith belief that such action is necessary to:
-
- To comply with a legal obligation
- To protect and defend the rights or property of domsPlace
- To prevent or investigate possible wrongdoing in connection with the Service
- To protect the personal safety of users of the Service or the public
- To protect against legal liability
-
-
- Security Of Data
- The security of your data is important to us, but remember that no method of transmission over the Internet, or method of electronic storage is 100% secure. While we strive to use commercially acceptable means to protect your Personal Data, we cannot guarantee its absolute security.
-
- Service Providers
- We may employ third party companies and individuals to facilitate our Service ("Service Providers"), to provide the Service on our behalf, to perform Service-related services or to assist us in analyzing how our Service is used.
- These third parties have access to your Personal Data only to perform these tasks on our behalf and are obligated not to disclose or use it for any other purpose.
-
- Analytics
- We may use third-party Service Providers to monitor and analyze the use of our Service.
-
-
- Google Analytics
- Google Analytics is a web analytics service offered by Google that tracks and reports website traffic. Google uses the data collected to track and monitor the use of our Service. This data is shared with other Google services. Google may use the collected data to contextualize and personalize the ads of its own advertising network.
- You can opt-out of having made your activity on the Service available to Google Analytics by installing the Google Analytics opt-out browser add-on. The add-on prevents the Google Analytics JavaScript (ga.js, analytics.js, and dc.js) from sharing information with Google Analytics about visits activity.
- For more information on the privacy practices of Google, please visit the Google Privacy & Terms web page: https://policies.google.com/privacy?hl=en
-
-
-
-
-
- Links To Other Sites
- Our Service may contain links to other sites that are not operated by us. If you click on a third party link, you will be directed to that third party's site. We strongly advise you to review the Privacy Policy of every site you visit.
- We have no control over and assume no responsibility for the content, privacy policies or practices of any third party sites or services.
-
-
- Children's Privacy
- Our Service does not address anyone under the age of 18 ("Children").
- We do not knowingly collect personally identifiable information from anyone under the age of 18. If you are a parent or guardian and you are aware that your Children has provided us with Personal Data, please contact us. If we become aware that we have collected Personal Data from children without verification of parental consent, we take steps to remove that information from our servers.
-
-
- Changes To This Privacy Policy
- We may update our Privacy Policy from time to time. We will notify you of any changes by posting the new Privacy Policy on this page.
- We will let you know via email and/or a prominent notice on our Service, prior to the change becoming effective and update the "effective date" at the top of this Privacy Policy.
- You are advised to review this Privacy Policy periodically for any changes. Changes to this Privacy Policy are effective when they are posted on this page.
-
-
- Contact Us
-
- If you have any questions about this Privacy Policy, please contact us
- by visiting this page on our
- website: https://domsplace.com/contact
-
-
-
-);
\ No newline at end of file
diff --git a/src/public/src/settings/all.ts b/src/public/src/settings/all.ts
deleted file mode 100644
index c33ef0c..0000000
--- a/src/public/src/settings/all.ts
+++ /dev/null
@@ -1,8 +0,0 @@
-export * from './styles/Animation';
-export * from './styles/Border';
-export * from './styles/Color';
-export * from './styles/Font';
-export * from './styles/Gutter';
-export * from './styles/Responsive';
-export * from './styles/Widget';
-export * from './styles/ZIndex';
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Animation.ts b/src/public/src/settings/styles/Animation.ts
deleted file mode 100644
index dbe30db..0000000
--- a/src/public/src/settings/styles/Animation.ts
+++ /dev/null
@@ -1,11 +0,0 @@
-export const Easings = {
- easeOut: 'cubic-bezier(0.165, 0.84, 0.44, 1)',
- easeInOut: 'cubic-bezier(0.77, 0, 0.175, 1)'
-}
-
-export const Durations = {
- timeShort: 0.2,
- timeMedium: 0.5,
- timeLong: 1,
- timeVeryLong: 3
-}
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Border.ts b/src/public/src/settings/styles/Border.ts
deleted file mode 100644
index fba43e7..0000000
--- a/src/public/src/settings/styles/Border.ts
+++ /dev/null
@@ -1,5 +0,0 @@
-import { Colors } from "./Color";
-
-export const Borders = {
- default: `0.2em solid ${Colors.default}`
-};
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Color.ts b/src/public/src/settings/styles/Color.ts
deleted file mode 100644
index 6dbf92c..0000000
--- a/src/public/src/settings/styles/Color.ts
+++ /dev/null
@@ -1,22 +0,0 @@
-export const Colors = {
- primary: '#B52756',
- primaryDark: '#370c1a',
- primaryLight: '#d53a6d',
-
- disabled: '#777',
- disabledDark: '#555',
-
- danger: '#e85a5a',
- dangerLight: '#ee8787',
- dangerDark: '#d71e1e',
-
- success: '#69d34a',
- successLight: '#ade79c',
- successDark: '#346925',
-
- default: '#000',
- defaultLight: '#333',
-
- background: '#000',
- text: '#FFF'
-}
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Font.ts b/src/public/src/settings/styles/Font.ts
deleted file mode 100644
index cd13773..0000000
--- a/src/public/src/settings/styles/Font.ts
+++ /dev/null
@@ -1,23 +0,0 @@
-export const Fonts = {
- body: "'Nanum Gothic', sans-serif",
- heading: "'Bitter', serif"
-};
-
-export const FontWeights = {
- body: 400,
- heading: 600
-};
-
-export const FontSizes = {
- default: '16px',
-
- title: '2.5rem',
- titleLarge: '3.5rem',
-
- heading1: '2.25rem',
- heading2: '1.75rem',
- heading3: '1.5rem',
- heading4: '1.25rem',
- heading5: '1.1rem',
- heading6: '0.9rem'
-}
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Gutter.ts b/src/public/src/settings/styles/Gutter.ts
deleted file mode 100644
index 2231121..0000000
--- a/src/public/src/settings/styles/Gutter.ts
+++ /dev/null
@@ -1,9 +0,0 @@
-export const Gutters = {
- extraSmall: '.25rem',
- small: '.5rem',
- medium: '1rem',
- large: '2rem',
- extraLarge: '3rem',
- extraExtraLarge: '4rem',
- extremeLarge: '12rem'
-};
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Responsive.ts b/src/public/src/settings/styles/Responsive.ts
deleted file mode 100644
index 1718adb..0000000
--- a/src/public/src/settings/styles/Responsive.ts
+++ /dev/null
@@ -1,27 +0,0 @@
-export const Sizes = {
- 'mobileSmall': 320,
- 'mobile': 375,
- 'mobileLarge': 414,
-
- 'tabletsmall': 600,
- 'tablet': 768,
- 'tabletLarge': 1024,
-
- 'laptopSmall': 1280,
- 'laptop': 1366,
- 'laptopLarge': 1600,
-
- 'desktopSmall': 1920,
- 'desktop': 2300,
- 'desktopLarge': 2560,
-
- /* Special */
- 'ultrawide': 3840
-}
-
-export const MediaQueries = {
- mobileUp: `@media screen and (min-width: ${Sizes.mobile}px)`,
- tabletUp: `@media screen and (min-width: ${Sizes.tablet}px)`,
- laptopUp: `@media screen and (min-width: ${Sizes.laptop}px)`,
- desktopUp: `@media screen and (min-width: ${Sizes.desktop}px)`,
-}
\ No newline at end of file
diff --git a/src/public/src/settings/styles/Widget.ts b/src/public/src/settings/styles/Widget.ts
deleted file mode 100644
index 42ad334..0000000
--- a/src/public/src/settings/styles/Widget.ts
+++ /dev/null
@@ -1,65 +0,0 @@
-import { Gutters } from "./Gutter";
-import { Borders } from "./Border";
-import { Colors } from "./Color";
-
-export type WidgetTheme = (
- 'default' | 'primary' | 'danger' | 'success'
-)
-
-export interface WidgetProps {
- disabled?:boolean;
- theme?:WidgetTheme;
-}
-
-export interface WidgetSettings {
- hover?:boolean;
-}
-
-export const WidgetStyles = ({
- disabled, theme
-}:WidgetProps, {
- hover
-}:WidgetSettings={}) => `
- display: inline-block;
- position: relative;
-
- font-weight: inherit;
- font-family: inherit;
- font-size: inherit;
- border: none;
- box-shadow: none;
- outline: none;
- appearance: none;
-
- padding: ${Gutters.medium} ${Gutters.large};
- border: ${Borders.default};
- color: ${Colors.text};
-
- margin-top: ${Gutters.small};
-
- ${disabled ? `
- background: ${Colors.disabled};
- border-color: ${Colors.disabled};
- color: ${Colors.disabledDark};
- cursor: not-allowed;
-
- ` : theme == 'primary' ? `
- background: ${Colors.primary};
- border-color: ${Colors.primaryDark};
- ${hover ? `&:hover { background: ${Colors.primaryLight}; }` : ''}
-
- ` : theme == 'danger' ? `
- background: ${Colors.danger};
- border-color: ${Colors.dangerDark};
- ${hover ? `&:hover { background: ${Colors.dangerLight}; }` : ''}
-
- ` : theme == 'success' ? `
- background: ${Colors.success};
- border-color: ${Colors.successDark};
- ${hover ? `&:hover { background: ${Colors.successLight}; }` : ''}
-
- ` : `
- ${hover ? `&:hover { border-color: ${Colors.defaultLight} }` : ''}
- background: transparent;
- `}
-`
\ No newline at end of file
diff --git a/src/public/src/settings/styles/ZIndex.ts b/src/public/src/settings/styles/ZIndex.ts
deleted file mode 100644
index 6e0e085..0000000
--- a/src/public/src/settings/styles/ZIndex.ts
+++ /dev/null
@@ -1,3 +0,0 @@
-export const ZIndex = {
- header: 10
-}
\ No newline at end of file
diff --git a/src/public/src/styles/all.tsx b/src/public/src/styles/all.tsx
deleted file mode 100644
index d3b0ede..0000000
--- a/src/public/src/styles/all.tsx
+++ /dev/null
@@ -1,7 +0,0 @@
-import { css } from "styled-components";
-
-export const AllElementStyles = css`
- * {
- box-sizing: border-box;
- }
-`;
\ No newline at end of file
diff --git a/src/public/src/styles/anchor.tsx b/src/public/src/styles/anchor.tsx
deleted file mode 100644
index 702b44b..0000000
--- a/src/public/src/styles/anchor.tsx
+++ /dev/null
@@ -1,10 +0,0 @@
-import { Colors } from "@settings/all";
-import { css } from "styled-components";
-
-export const AnchorStyles = css`
- a {
- text-decoration: inherit;
- color: ${Colors.primary};
- &:hover { text-decoration: underline; }
- }
-`;
\ No newline at end of file
diff --git a/src/public/src/styles/body.tsx b/src/public/src/styles/body.tsx
deleted file mode 100644
index 43c7fb1..0000000
--- a/src/public/src/styles/body.tsx
+++ /dev/null
@@ -1,18 +0,0 @@
-import { Colors, Fonts, FontWeights, FontSizes } from "@settings/all";
-import { css } from "styled-components";
-
-export const BodyStyles = css`
- body,html {
- margin: 0;
- padding: 0;
- }
-
-
- body {
- color: ${Colors.text};
- background: ${Colors.background};
- font-family: ${Fonts.body};
- font-weight: ${FontWeights.body};
- font-size: ${FontSizes.default};
- }
-`;
\ No newline at end of file
diff --git a/src/public/src/styles/index.tsx b/src/public/src/styles/index.tsx
deleted file mode 100644
index 71cb973..0000000
--- a/src/public/src/styles/index.tsx
+++ /dev/null
@@ -1,3 +0,0 @@
-export * from './all';
-export * from './body';
-export * from './anchor';
\ No newline at end of file
diff --git a/src/public/src/tools/styles.ts b/src/public/src/tools/styles.ts
deleted file mode 100644
index 1ec3ff0..0000000
--- a/src/public/src/tools/styles.ts
+++ /dev/null
@@ -1,9 +0,0 @@
-import { css } from "styled-components";
-
-export const StylePlaceholder = (content:string) => css`
- &::placeholder,
- &::-webkit-input-placeholder,
- &:-ms-input-placeholder {
- ${content}
- }
-`;
\ No newline at end of file
diff --git a/src/public/tsconfig.json b/src/public/tsconfig.json
deleted file mode 100644
index 9cb6484..0000000
--- a/src/public/tsconfig.json
+++ /dev/null
@@ -1,36 +0,0 @@
-{
- "compilerOptions": {
- "target": "es5",
- "lib": [
- "dom",
- "dom.iterable",
- "esnext"
- ],
- "allowJs": true,
- "skipLibCheck": true,
- "esModuleInterop": true,
- "allowSyntheticDefaultImports": true,
- "strict": true,
- "forceConsistentCasingInFileNames": true,
- "module": "esnext",
- "resolveJsonModule": true,
- "isolatedModules": true,
- "noEmit": true,
- "jsx": "preserve",
- "moduleResolution": "node",
- "baseUrl": "src",
- "paths": {
- "@api/*": [ "api/*" ],
- "@components/*": [ "components/*" ],
- "@objects/*": [ "objects/*" ],
- "@settings/*": [ "settings/*" ],
- "@styles/*": [ "styles/*" ],
- "@pages/*": [ "pages/*" ],
- "@tools/*": [ "tools/*" ],
- "@assets/*": [ "assets/*" ]
- }
- },
- "include": [
- "src"
- ]
-}
\ No newline at end of file
diff --git a/src/serve.ts b/src/serve.ts
new file mode 100644
index 0000000..8c4ffff
--- /dev/null
+++ b/src/serve.ts
@@ -0,0 +1,20 @@
+// Copyright (c) 2021 Dominic Masters
+//
+// This software is released under the MIT License.
+// https://opensource.org/licenses/MIT
+
+import * as express from 'express';
+import { pageCompileAll, PATH_OUT } from './compiler';
+
+const app = express()
+const port = 80;
+
+app.use((req, res, next) => {
+ pageCompileAll();
+ next();
+})
+app.use(express.static(PATH_OUT));
+
+app.listen(port, () => {
+ console.log(`Example app listening at http://localhost:${port}`)
+})
\ No newline at end of file
diff --git a/yarn.lock b/yarn.lock
index 4cded13..f54aac4 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -2,27 +2,489 @@
# yarn lockfile v1
-"@types/prop-types@*":
- version "15.7.3"
- resolved "https://registry.yarnpkg.com/@types/prop-types/-/prop-types-15.7.3.tgz#2ab0d5da2e5815f94b0b9d4b95d1e5f243ab2ca7"
- integrity sha512-KfRL3PuHmqQLOG+2tGpRO26Ctg+Cq1E01D2DMriKEATHgWLfeNDmq9e29Q9WIky0dQ3NPkd1mzYH8Lm936Z9qw==
-
-"@types/react-syntax-highlighter@^11.0.4":
- version "11.0.4"
- resolved "https://registry.yarnpkg.com/@types/react-syntax-highlighter/-/react-syntax-highlighter-11.0.4.tgz#d86d17697db62f98046874f62fdb3e53a0bbc4cd"
- integrity sha512-9GfTo3a0PHwQeTVoqs0g5bS28KkSY48pp5659wA+Dp4MqceDEa8EHBqrllJvvtyusszyJhViUEap0FDvlk/9Zg==
+"@types/body-parser@*":
+ version "1.19.0"
+ resolved "https://registry.yarnpkg.com/@types/body-parser/-/body-parser-1.19.0.tgz#0685b3c47eb3006ffed117cdd55164b61f80538f"
+ integrity sha512-W98JrE0j2K78swW4ukqMleo8R7h/pFETjM2DQ90MF6XK2i4LO4W3gQ71Lt4w3bfm2EvVSyWHplECvB5sK22yFQ==
dependencies:
- "@types/react" "*"
+ "@types/connect" "*"
+ "@types/node" "*"
-"@types/react@*":
- version "16.9.19"
- resolved "https://registry.yarnpkg.com/@types/react/-/react-16.9.19.tgz#c842aa83ea490007d29938146ff2e4d9e4360c40"
- integrity sha512-LJV97//H+zqKWMms0kvxaKYJDG05U2TtQB3chRLF8MPNs+MQh/H1aGlyDUxjaHvu08EAGerdX2z4LTBc7ns77A==
+"@types/connect@*":
+ version "3.4.34"
+ resolved "https://registry.yarnpkg.com/@types/connect/-/connect-3.4.34.tgz#170a40223a6d666006d93ca128af2beb1d9b1901"
+ integrity sha512-ePPA/JuI+X0vb+gSWlPKOY0NdNAie/rPUqX2GUPpbZwiKTkSPhjXWuee47E4MtE54QVzGCQMQkAL6JhV2E1+cQ==
dependencies:
- "@types/prop-types" "*"
- csstype "^2.2.0"
+ "@types/node" "*"
-csstype@^2.2.0:
- version "2.6.8"
- resolved "https://registry.yarnpkg.com/csstype/-/csstype-2.6.8.tgz#0fb6fc2417ffd2816a418c9336da74d7f07db431"
- integrity sha512-msVS9qTuMT5zwAGCVm4mxfrZ18BNc6Csd0oJAtiFMZ1FAx1CCvy2+5MDmYoix63LM/6NDbNtodCiGYGmFgO0dA==
+"@types/express-serve-static-core@^4.17.18":
+ version "4.17.19"
+ resolved "https://registry.yarnpkg.com/@types/express-serve-static-core/-/express-serve-static-core-4.17.19.tgz#00acfc1632e729acac4f1530e9e16f6dd1508a1d"
+ integrity sha512-DJOSHzX7pCiSElWaGR8kCprwibCB/3yW6vcT8VG3P0SJjnv19gnWG/AZMfM60Xj/YJIp/YCaDHyvzsFVeniARA==
+ dependencies:
+ "@types/node" "*"
+ "@types/qs" "*"
+ "@types/range-parser" "*"
+
+"@types/express@^4.17.11":
+ version "4.17.11"
+ resolved "https://registry.yarnpkg.com/@types/express/-/express-4.17.11.tgz#debe3caa6f8e5fcda96b47bd54e2f40c4ee59545"
+ integrity sha512-no+R6rW60JEc59977wIxreQVsIEOAYwgCqldrA/vkpCnbD7MqTefO97lmoBe4WE0F156bC4uLSP1XHDOySnChg==
+ dependencies:
+ "@types/body-parser" "*"
+ "@types/express-serve-static-core" "^4.17.18"
+ "@types/qs" "*"
+ "@types/serve-static" "*"
+
+"@types/mime@^1":
+ version "1.3.2"
+ resolved "https://registry.yarnpkg.com/@types/mime/-/mime-1.3.2.tgz#93e25bf9ee75fe0fd80b594bc4feb0e862111b5a"
+ integrity sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw==
+
+"@types/node@*":
+ version "14.14.35"
+ resolved "https://registry.yarnpkg.com/@types/node/-/node-14.14.35.tgz#42c953a4e2b18ab931f72477e7012172f4ffa313"
+ integrity sha512-Lt+wj8NVPx0zUmUwumiVXapmaLUcAk3yPuHCFVXras9k5VT9TdhJqKqGVUQCD60OTMCl0qxJ57OiTL0Mic3Iag==
+
+"@types/qs@*":
+ version "6.9.6"
+ resolved "https://registry.yarnpkg.com/@types/qs/-/qs-6.9.6.tgz#df9c3c8b31a247ec315e6996566be3171df4b3b1"
+ integrity sha512-0/HnwIfW4ki2D8L8c9GVcG5I72s9jP5GSLVF0VIXDW00kmIpA6O33G7a8n59Tmh7Nz0WUC3rSb7PTY/sdW2JzA==
+
+"@types/range-parser@*":
+ version "1.2.3"
+ resolved "https://registry.yarnpkg.com/@types/range-parser/-/range-parser-1.2.3.tgz#7ee330ba7caafb98090bece86a5ee44115904c2c"
+ integrity sha512-ewFXqrQHlFsgc09MK5jP5iR7vumV/BYayNC6PgJO2LPe8vrnNFyjQjSppfEngITi0qvfKtzFvgKymGheFM9UOA==
+
+"@types/serve-static@*":
+ version "1.13.9"
+ resolved "https://registry.yarnpkg.com/@types/serve-static/-/serve-static-1.13.9.tgz#aacf28a85a05ee29a11fb7c3ead935ac56f33e4e"
+ integrity sha512-ZFqF6qa48XsPdjXV5Gsz0Zqmux2PerNd3a/ktL45mHpa19cuMi/cL8tcxdAx497yRh+QtYPuofjT9oWw9P7nkA==
+ dependencies:
+ "@types/mime" "^1"
+ "@types/node" "*"
+
+accepts@~1.3.7:
+ version "1.3.7"
+ resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.7.tgz#531bc726517a3b2b41f850021c6cc15eaab507cd"
+ integrity sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==
+ dependencies:
+ mime-types "~2.1.24"
+ negotiator "0.6.2"
+
+arg@^4.1.0:
+ version "4.1.3"
+ resolved "https://registry.yarnpkg.com/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089"
+ integrity sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==
+
+array-flatten@1.1.1:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2"
+ integrity sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=
+
+body-parser@1.19.0:
+ version "1.19.0"
+ resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.19.0.tgz#96b2709e57c9c4e09a6fd66a8fd979844f69f08a"
+ integrity sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==
+ dependencies:
+ bytes "3.1.0"
+ content-type "~1.0.4"
+ debug "2.6.9"
+ depd "~1.1.2"
+ http-errors "1.7.2"
+ iconv-lite "0.4.24"
+ on-finished "~2.3.0"
+ qs "6.7.0"
+ raw-body "2.4.0"
+ type-is "~1.6.17"
+
+buffer-from@^1.0.0:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz#32713bc028f75c02fdb710d7c7bcec1f2c6070ef"
+ integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==
+
+bytes@3.1.0:
+ version "3.1.0"
+ resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.0.tgz#f6cf7933a360e0588fa9fde85651cdc7f805d1f6"
+ integrity sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==
+
+content-disposition@0.5.3:
+ version "0.5.3"
+ resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.3.tgz#e130caf7e7279087c5616c2007d0485698984fbd"
+ integrity sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==
+ dependencies:
+ safe-buffer "5.1.2"
+
+content-type@~1.0.4:
+ version "1.0.4"
+ resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.4.tgz#e138cc75e040c727b1966fe5e5f8c9aee256fe3b"
+ integrity sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==
+
+cookie-signature@1.0.6:
+ version "1.0.6"
+ resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c"
+ integrity sha1-4wOogrNCzD7oylE6eZmXNNqzriw=
+
+cookie@0.4.0:
+ version "0.4.0"
+ resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.0.tgz#beb437e7022b3b6d49019d088665303ebe9c14ba"
+ integrity sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==
+
+create-require@^1.1.0:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/create-require/-/create-require-1.1.1.tgz#c1d7e8f1e5f6cfc9ff65f9cd352d37348756c333"
+ integrity sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==
+
+debug@2.6.9:
+ version "2.6.9"
+ resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
+ integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
+ dependencies:
+ ms "2.0.0"
+
+depd@~1.1.2:
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9"
+ integrity sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=
+
+destroy@~1.0.4:
+ version "1.0.4"
+ resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80"
+ integrity sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=
+
+diff@^4.0.1:
+ version "4.0.2"
+ resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d"
+ integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==
+
+ee-first@1.1.1:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
+ integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=
+
+encodeurl@~1.0.2:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
+ integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=
+
+escape-html@~1.0.3:
+ version "1.0.3"
+ resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988"
+ integrity sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=
+
+etag@~1.8.1:
+ version "1.8.1"
+ resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
+ integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=
+
+express@^4.17.1:
+ version "4.17.1"
+ resolved "https://registry.yarnpkg.com/express/-/express-4.17.1.tgz#4491fc38605cf51f8629d39c2b5d026f98a4c134"
+ integrity sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==
+ dependencies:
+ accepts "~1.3.7"
+ array-flatten "1.1.1"
+ body-parser "1.19.0"
+ content-disposition "0.5.3"
+ content-type "~1.0.4"
+ cookie "0.4.0"
+ cookie-signature "1.0.6"
+ debug "2.6.9"
+ depd "~1.1.2"
+ encodeurl "~1.0.2"
+ escape-html "~1.0.3"
+ etag "~1.8.1"
+ finalhandler "~1.1.2"
+ fresh "0.5.2"
+ merge-descriptors "1.0.1"
+ methods "~1.1.2"
+ on-finished "~2.3.0"
+ parseurl "~1.3.3"
+ path-to-regexp "0.1.7"
+ proxy-addr "~2.0.5"
+ qs "6.7.0"
+ range-parser "~1.2.1"
+ safe-buffer "5.1.2"
+ send "0.17.1"
+ serve-static "1.14.1"
+ setprototypeof "1.1.1"
+ statuses "~1.5.0"
+ type-is "~1.6.18"
+ utils-merge "1.0.1"
+ vary "~1.1.2"
+
+finalhandler@~1.1.2:
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.2.tgz#b7e7d000ffd11938d0fdb053506f6ebabe9f587d"
+ integrity sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==
+ dependencies:
+ debug "2.6.9"
+ encodeurl "~1.0.2"
+ escape-html "~1.0.3"
+ on-finished "~2.3.0"
+ parseurl "~1.3.3"
+ statuses "~1.5.0"
+ unpipe "~1.0.0"
+
+forwarded@~0.1.2:
+ version "0.1.2"
+ resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.2.tgz#98c23dab1175657b8c0573e8ceccd91b0ff18c84"
+ integrity sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=
+
+fresh@0.5.2:
+ version "0.5.2"
+ resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7"
+ integrity sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=
+
+http-errors@1.7.2:
+ version "1.7.2"
+ resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.2.tgz#4f5029cf13239f31036e5b2e55292bcfbcc85c8f"
+ integrity sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==
+ dependencies:
+ depd "~1.1.2"
+ inherits "2.0.3"
+ setprototypeof "1.1.1"
+ statuses ">= 1.5.0 < 2"
+ toidentifier "1.0.0"
+
+http-errors@~1.7.2:
+ version "1.7.3"
+ resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.3.tgz#6c619e4f9c60308c38519498c14fbb10aacebb06"
+ integrity sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==
+ dependencies:
+ depd "~1.1.2"
+ inherits "2.0.4"
+ setprototypeof "1.1.1"
+ statuses ">= 1.5.0 < 2"
+ toidentifier "1.0.0"
+
+iconv-lite@0.4.24:
+ version "0.4.24"
+ resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
+ integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
+ dependencies:
+ safer-buffer ">= 2.1.2 < 3"
+
+inherits@2.0.3:
+ version "2.0.3"
+ resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de"
+ integrity sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=
+
+inherits@2.0.4:
+ version "2.0.4"
+ resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
+ integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+
+ipaddr.js@1.9.1:
+ version "1.9.1"
+ resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3"
+ integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==
+
+make-error@^1.1.1:
+ version "1.3.6"
+ resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2"
+ integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==
+
+media-typer@0.3.0:
+ version "0.3.0"
+ resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748"
+ integrity sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=
+
+merge-descriptors@1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61"
+ integrity sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=
+
+methods@~1.1.2:
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee"
+ integrity sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=
+
+mime-db@1.46.0:
+ version "1.46.0"
+ resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.46.0.tgz#6267748a7f799594de3cbc8cde91def349661cee"
+ integrity sha512-svXaP8UQRZ5K7or+ZmfNhg2xX3yKDMUzqadsSqi4NCH/KomcH75MAMYAGVlvXn4+b/xOPhS3I2uHKRUzvjY7BQ==
+
+mime-types@~2.1.24:
+ version "2.1.29"
+ resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.29.tgz#1d4ab77da64b91f5f72489df29236563754bb1b2"
+ integrity sha512-Y/jMt/S5sR9OaqteJtslsFZKWOIIqMACsJSiHghlCAyhf7jfVYjKBmLiX8OgpWeW+fjJ2b+Az69aPFPkUOY6xQ==
+ dependencies:
+ mime-db "1.46.0"
+
+mime@1.6.0:
+ version "1.6.0"
+ resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1"
+ integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==
+
+ms@2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8"
+ integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=
+
+ms@2.1.1:
+ version "2.1.1"
+ resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.1.tgz#30a5864eb3ebb0a66f2ebe6d727af06a09d86e0a"
+ integrity sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==
+
+negotiator@0.6.2:
+ version "0.6.2"
+ resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.2.tgz#feacf7ccf525a77ae9634436a64883ffeca346fb"
+ integrity sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==
+
+on-finished@~2.3.0:
+ version "2.3.0"
+ resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947"
+ integrity sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=
+ dependencies:
+ ee-first "1.1.1"
+
+parseurl@~1.3.3:
+ version "1.3.3"
+ resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4"
+ integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==
+
+path-to-regexp@0.1.7:
+ version "0.1.7"
+ resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c"
+ integrity sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=
+
+proxy-addr@~2.0.5:
+ version "2.0.6"
+ resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.6.tgz#fdc2336505447d3f2f2c638ed272caf614bbb2bf"
+ integrity sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==
+ dependencies:
+ forwarded "~0.1.2"
+ ipaddr.js "1.9.1"
+
+qs@6.7.0:
+ version "6.7.0"
+ resolved "https://registry.yarnpkg.com/qs/-/qs-6.7.0.tgz#41dc1a015e3d581f1621776be31afb2876a9b1bc"
+ integrity sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==
+
+range-parser@~1.2.1:
+ version "1.2.1"
+ resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031"
+ integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==
+
+raw-body@2.4.0:
+ version "2.4.0"
+ resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.4.0.tgz#a1ce6fb9c9bc356ca52e89256ab59059e13d0332"
+ integrity sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==
+ dependencies:
+ bytes "3.1.0"
+ http-errors "1.7.2"
+ iconv-lite "0.4.24"
+ unpipe "1.0.0"
+
+safe-buffer@5.1.2:
+ version "5.1.2"
+ resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d"
+ integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==
+
+"safer-buffer@>= 2.1.2 < 3":
+ version "2.1.2"
+ resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a"
+ integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==
+
+send@0.17.1:
+ version "0.17.1"
+ resolved "https://registry.yarnpkg.com/send/-/send-0.17.1.tgz#c1d8b059f7900f7466dd4938bdc44e11ddb376c8"
+ integrity sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==
+ dependencies:
+ debug "2.6.9"
+ depd "~1.1.2"
+ destroy "~1.0.4"
+ encodeurl "~1.0.2"
+ escape-html "~1.0.3"
+ etag "~1.8.1"
+ fresh "0.5.2"
+ http-errors "~1.7.2"
+ mime "1.6.0"
+ ms "2.1.1"
+ on-finished "~2.3.0"
+ range-parser "~1.2.1"
+ statuses "~1.5.0"
+
+serve-static@1.14.1:
+ version "1.14.1"
+ resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.14.1.tgz#666e636dc4f010f7ef29970a88a674320898b2f9"
+ integrity sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==
+ dependencies:
+ encodeurl "~1.0.2"
+ escape-html "~1.0.3"
+ parseurl "~1.3.3"
+ send "0.17.1"
+
+setprototypeof@1.1.1:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.1.1.tgz#7e95acb24aa92f5885e0abef5ba131330d4ae683"
+ integrity sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==
+
+source-map-support@^0.5.17:
+ version "0.5.19"
+ resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.19.tgz#a98b62f86dcaf4f67399648c085291ab9e8fed61"
+ integrity sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==
+ dependencies:
+ buffer-from "^1.0.0"
+ source-map "^0.6.0"
+
+source-map@^0.6.0:
+ version "0.6.1"
+ resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263"
+ integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==
+
+"statuses@>= 1.5.0 < 2", statuses@~1.5.0:
+ version "1.5.0"
+ resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c"
+ integrity sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=
+
+toidentifier@1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.0.tgz#7e1be3470f1e77948bc43d94a3c8f4d7752ba553"
+ integrity sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==
+
+ts-node@^9.1.1:
+ version "9.1.1"
+ resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-9.1.1.tgz#51a9a450a3e959401bda5f004a72d54b936d376d"
+ integrity sha512-hPlt7ZACERQGf03M253ytLY3dHbGNGrAq9qIHWUY9XHYl1z7wYngSr3OQ5xmui8o2AaxsONxIzjafLUiWBo1Fg==
+ dependencies:
+ arg "^4.1.0"
+ create-require "^1.1.0"
+ diff "^4.0.1"
+ make-error "^1.1.1"
+ source-map-support "^0.5.17"
+ yn "3.1.1"
+
+type-is@~1.6.17, type-is@~1.6.18:
+ version "1.6.18"
+ resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131"
+ integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==
+ dependencies:
+ media-typer "0.3.0"
+ mime-types "~2.1.24"
+
+typescript@^4.2.3:
+ version "4.2.3"
+ resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.2.3.tgz#39062d8019912d43726298f09493d598048c1ce3"
+ integrity sha512-qOcYwxaByStAWrBf4x0fibwZvMRG+r4cQoTjbPtUlrWjBHbmCAww1i448U0GJ+3cNNEtebDteo/cHOR3xJ4wEw==
+
+unpipe@1.0.0, unpipe@~1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec"
+ integrity sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=
+
+utils-merge@1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713"
+ integrity sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=
+
+vary@~1.1.2:
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc"
+ integrity sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=
+
+yn@3.1.1:
+ version "3.1.1"
+ resolved "https://registry.yarnpkg.com/yn/-/yn-3.1.1.tgz#1e87401a09d767c1d5eab26a6e4c185182d2eb50"
+ integrity sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==