Skip to content

Commit

Permalink
feat: mvp (#1)
Browse files Browse the repository at this point in the history
  • Loading branch information
scissorsneedfoodtoo committed Jul 7, 2021
1 parent 8f7e86c commit 11784cf
Show file tree
Hide file tree
Showing 90 changed files with 11,786 additions and 57 deletions.
26 changes: 26 additions & 0 deletions .editorconfig
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
# http://editorconfig.org

root = true

[*]
charset = utf-8
indent_style = space
indent_size = 4
end_of_line = lf
insert_final_newline = true
trim_trailing_whitespace = true

[*.hbs]
insert_final_newline = false

[*.json]
indent_size = 2

[*.md]
trim_trailing_whitespace = false

[*.{yml,yaml}]
indent_size = 2

[Makefile]
indent_style = tab
360 changes: 360 additions & 0 deletions .eleventy.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,360 @@
require("dotenv").config();

const htmlMin = require("./utils/transforms/html-min");
const cssMin = require("./utils/transforms/css-min");
const jsMin = require("./utils/transforms/js-min");
const { readFileSync, readdirSync, writeFileSync } = require("fs");
const { basename, extname } = require("path");
const Image = require("@11ty/eleventy-img");
const pluginRSS = require("@11ty/eleventy-plugin-rss");
const { api } = require("./utils/ghost-api");
const i18next = require("./i18n/config");
const dayjs = require("./utils/dayjs");
const cacheBuster = require("@mightyplow/eleventy-plugin-cache-buster");
const { settings } = require('./utils/ghost-settings');
const { escape } = require('lodash');

module.exports = function(config) {
// Minify HTML
config.addTransform("htmlMin", htmlMin);

// Minify inline CSS
config.addFilter("cssMin", cssMin);

// Minify inline JS
config.addNunjucksAsyncFilter("jsMin", jsMin);

// Allow passthrough for styles, scripts, and images
config.addPassthroughCopy({'./src/_includes/assets': './assets/'});

// Minify CSS
config.on('afterBuild', () => {
const path = './dist/assets/css';
const cssFiles = readdirSync(path);

cssFiles.forEach(filename => {
const fullPath = `${path}/${filename}`;
const content = readFileSync(fullPath);

writeFileSync(fullPath, cssMin(content));
});
});

// Basic cache busting
config.addPlugin(
cacheBuster({
outputDirectory: './dist',
})
);

// Assist RSS feed template
config.addPlugin(pluginRSS);

// Copy images over from Ghost
function imageShortcode(src, cls, alt, sizes, widths, index) {
const imageFormats = ["webp"];
const imageExtension = extname(src);
const imageName = basename(src, imageExtension).split('?')[0]; // strip off url params, if any
const options = {
widths: widths,
formats: imageFormats,
outputDir: "./dist/assets/images/",
filenameFormat: function (id, src, width, format, options) {
return `${imageName}-${width}w.${format}`
}
}

// generate images, while this is async we don’t wait
Image(src, options);

return `
<img
${index === 0 ? `rel="preload" as="image"` : ''}
${(cls.includes('lazyload') && index > 0) ? 'data-srcset' : 'srcset'}="${widths.map(width => `/assets/images/${imageName}-${width}w.webp ${width}w`).join()}"
sizes="${sizes.replace(/\s+/g, ' ').trim()}"
${(cls.includes('lazyload') && index > 0) ? 'data-src' : 'src'}="/assets/images/${imageName}-${widths[0]}w.webp"
class="${index === 0 ? cls.replace('lazyload', '') : cls}"
alt="${alt}"
onerror="this.style.display='none'"
/>
`;
}

config.addNunjucksShortcode("image", imageShortcode);

// Copy images over from Ghost
function featureImageShortcode(src, alt, sizes, widths) {
const imageFormats = ["webp"];
const imageExtension = extname(src);
const imageName = basename(src, imageExtension).split('?')[0]; // strip off url params, if any
const options = {
widths: widths,
formats: imageFormats,
outputDir: "./dist/assets/images/",
filenameFormat: function (id, src, width, format, options) {
return `${imageName}-${width}w.${format}`
}
}

// generate images, while this is async we don’t wait
Image(src, options);

return `
<picture>
<source
media="(max-width: 700px)"
sizes="1px"
srcset=" 1w"
/>
<source
media="(min-width: 701px)"
sizes="${sizes.replace(/\s+/g, ' ').trim()}"
srcset="${widths.map(width => `/assets/images/${imageName}-${width}w.webp ${width}w`).join()}"
/>
<img
onerror="this.style.display='none'"
src="/assets/images/${imageName}-${widths[0]}w.webp"
alt="${alt}"
>
</picture>
`;
}

config.addNunjucksShortcode("featureImage", featureImageShortcode);

// Date and time shortcodes
function publishedDateShortcode(dateStr) {
return dayjs(dateStr).format('LL');
}

config.addNunjucksShortcode("publishedDate", publishedDateShortcode);

function timeAgoShortcode(dateStr) {
return dayjs().to(dayjs(dateStr));
}

config.addNunjucksShortcode("timeAgo", timeAgoShortcode);

function translateShortcode(key, data) {
return i18next.t(key, { ...data });
}

config.addNunjucksShortcode("t", translateShortcode);

// Special handling for full stops
function fullStopHandlerShortcode(siteLang) {
const ideographicFullStopLanguageCodes = ['zh', 'zh-cn'];

return ideographicFullStopLanguageCodes.includes(siteLang) ? '。' : '.';
}

config.addNunjucksShortcode("fullStopHandler", fullStopHandlerShortcode);

config.addFilter("getReadingTime", text => {
const wordsPerMinute = 200;
const numberOfWords = text.split(/\s/g).length;
return Math.ceil(numberOfWords / wordsPerMinute);
});

// Date formatting filter
config.addFilter("htmlDateString", dateObj => {
return new Date(dateObj).toISOString().split("T")[0];
});

// Format dates for RSS feed
const buildDateFormatterShortcode = dateStr => {
const dateObj = dateStr ? new Date(dateStr) : new Date();
return dateObj.toUTCString();
}

config.addNunjucksShortcode("buildDateFormatter", buildDateFormatterShortcode);

config.addFilter("commentsEnabled", tagsArr => {
return !tagsArr.map(tag => tag.name).includes('#disable-comments');
});

// This counts on all images, including the site logo, being stored like on Ghost with the
// same directory structure
const domainReplacer = url => url.replace(process.env.GHOST_API_URL, process.env.SITE_URL);

// Mimic Ghost/Handlebars escaping
// raw: & < > " ' ` =
// html-escaped: &amp; &lt; &gt; &quot; &#x27; &#x60; &#x3D;
const fullEscaper = s => escape(s)
.replace(/&#39;/g, '&#x27;')
.replace(/`/g, '&#x60;')
.replace(/=/g, '&#x3D;');

config.addNunjucksShortcode("fullEscaper", fullEscaper);

async function createJsonLdShortcode(type, data) {
// Main site settings from Ghost API
let { url, logo, cover_image, image_dimensions } = await settings;
url = `${url}/`
const typeMap = {
index: 'WebSite',
article: 'Article',
author: 'Person',
tag: 'Series'
}
const baseData = {
"@context": "https://schema.org",
"@type": typeMap[type],
publisher: {
"@type": "Organization",
name: "freeCodeCamp.org",
url: url,
logo: {
"@type": "ImageObject",
url: domainReplacer(logo),
width: image_dimensions.logo.width,
height: image_dimensions.logo.height
}
},
image: {
"@type": "ImageObject",
url: domainReplacer(cover_image),
width: image_dimensions.cover_image.width,
height: image_dimensions.cover_image.height
},
url: url,
mainEntityOfPage: {
"@type": "WebPage",
"@id": url
}
}
const returnData = {...baseData};

// Would probably be better to look up image dimensions in ghost.js,
// to prevent looking up the same dimensions for each author image.
// Could also keep a map of article or page feature_images, if we want
// to calculate all that there, too
const createImageObj = (url, obj) => {
let { width, height } = obj;

return {
"@type": "ImageObject",
url,
width,
height
}
}

// Conditionally set other properties based on
// objects passed to shortcodes
const createAuthorObj = (primaryAuthor) => {
const {
name,
profile_image,
image_dimensions,
website,
twitter,
facebook
} = primaryAuthor;
const authorObj = {
'@type': 'Person',
name,
url: domainReplacer(url), // check again later when using slugs throughout template and leaving URLs untouched
sameAs: [
website ? fullEscaper(website) : null,
facebook ? `https://www.facebook.com/${facebook}` : null,
twitter ? twitter.replace('@', 'https://twitter.com/') : null
].filter(url => url)
}

if (profile_image) {
authorObj.image = createImageObj(profile_image, image_dimensions.profile_image);
}

return authorObj;
}

// Remove first slash from path
if (data.path) returnData.url += data.path.substring(1);

if (data.description) returnData.description = fullEscaper(data.description);

if (type === 'article') {
if (data.published_at) returnData.datePublished = new Date(data.published_at).toISOString();
if (data.updated_at) returnData.dateModified = new Date(data.updated_at).toISOString();
if (data.tags && data.tags.length > 1) {
// Filter out internal Ghost tags
const keywords = data.tags.map(tag => tag.name).filter(keyword => !keyword.startsWith('#'));

returnData.keywords = keywords.length === 1 ? keywords[0] : keywords;
};
if (data.excerpt) returnData.description = fullEscaper(data.excerpt);
if (data.title) returnData.headline = fullEscaper(data.title);

if (data.feature_image) {
returnData.image = await createImageObj(data.feature_image, data.image_dimensions.feature_image);
}

returnData.author = await createAuthorObj(data.primary_author);
}

// Handle images for both types
if (type === 'tag' || type === 'author') {
if (data.cover_image) {
returnData.image = createImageObj(data.cover_image, data.image_dimensions.cover_image);
} else if (data.feature_image) {
returnData.image = createImageObj(data.feature_image, data.image_dimensions.feature_image);
} else {
delete returnData.image;
}
}

if (type === 'tag') {
if (data.cover_image) returnData.image = createImageObj(data.cover_image, data.image_dimensions.cover_image);
returnData.name = data.name;
}

if (type === 'author') {
// This schema type is the only one without publisher info
delete returnData.publisher;
const authorObj = await createAuthorObj(data);

returnData.sameAs = authorObj.sameAs;
returnData.name = fullEscaper(authorObj.name);
}

return JSON.stringify(returnData, null, '\t'); // Pretty print for testing
// return JSON.stringify(returnData);
}

config.addNunjucksAsyncShortcode("createJsonLd", createJsonLdShortcode);

// Don't ignore the same files ignored in the git repo
config.setUseGitIgnore(false);

// Display 404 page in BrowserSnyc
config.setBrowserSyncConfig({
callbacks: {
ready: (err, bs) => {
const content_404 = readFileSync("dist/404.html");

bs.addMiddleware("*", (req, res) => {
res.writeHead(404, { "Content-Type": "text/html; charset=UTF-8" });

// Provides the 404 content without redirect.
res.write(content_404);
res.end();
});
}
}
});

// Eleventy configuration
return {
dir: {
input: "src",
output: "dist"
},

// Files read by Eleventy, add as needed
templateFormats: ["css", "njk", "md", "txt", "hbs"],
htmlTemplateEngine: "njk",
markdownTemplateEngine: "njk",
passthroughFileCopy: true
};
};
Loading

0 comments on commit 11784cf

Please sign in to comment.