programmingfonts/index.js

427 lines
14 KiB
JavaScript
Raw Normal View History

/* global CodeMirror window document Set */
2017-09-02 21:26:30 +02:00
/* eslint-disable no-implicit-globals */
2017-07-28 23:07:09 +02:00
2017-01-20 17:59:59 +01:00
// CodeMirror
2019-06-10 10:18:45 +02:00
var editor = CodeMirror.fromTextArea(document.getElementById('code'), {
lineNumbers: true,
styleActiveLine: true,
matchBrackets: true,
theme: 'pastel-on-dark',
2022-12-28 16:05:30 +01:00
lineWrapping: true,
2017-01-20 17:59:59 +01:00
});
// CodeMirror theme selector
2019-06-10 10:18:45 +02:00
var input = document.getElementById('select-theme');
2017-09-02 21:26:30 +02:00
var font_data;
2019-06-10 14:58:21 +02:00
var filters = {
2022-12-28 16:05:30 +01:00
style: false,
rendering: false,
liga: false,
zerostyle: false,
author: 'all',
name: '',
2019-06-10 14:58:21 +02:00
};
2022-12-28 16:05:30 +01:00
const selectTheme = () => {
2019-06-10 10:18:45 +02:00
var theme = 'monokai';
2017-09-02 21:26:30 +02:00
2019-06-10 10:18:45 +02:00
if (input.selectedIndex > -1) {
theme = input.options[input.selectedIndex].innerHTML;
}
editor.setOption('theme', theme);
2022-12-28 16:05:30 +01:00
document.cookie = `theme=${theme};max-age=172800`;
2022-12-28 16:46:17 +01:00
};
2017-01-20 17:59:59 +01:00
// ProgrammingFonts font selector
2022-12-28 16:05:30 +01:00
const selectFont = () => {
2019-06-10 10:18:45 +02:00
var font = window.location.hash.substring(1);
var status_msg = document.querySelector('footer .subtitle');
var code_mirror = document.querySelector('.CodeMirror');
2019-06-10 10:18:45 +02:00
2022-12-28 16:05:30 +01:00
if (!font) {
font = 'cartograph';
status_msg.innerHTML = 'Test drive all the programming fonts!';
} else if (typeof font_data !== 'undefined') {
2022-12-28 16:46:17 +01:00
status_msg.innerHTML = `Test drive <a rel="external" href="${font_data[font].website}">${font_data[font].name}!</a>`;
2019-06-10 10:18:45 +02:00
}
if (typeof font_data !== 'undefined' && font_data[font].rendering === 'bitmap') {
code_mirror.classList.add('no-smooth');
} else {
code_mirror.classList.remove('no-smooth');
}
2019-06-10 10:18:45 +02:00
if (font === 'input') {
code_mirror.style.fontFamily = 'Input Mono, monospace';
2022-12-28 16:46:17 +01:00
code_mirror.querySelectorAll('pre, textarea').forEach((element) => {
element.style.fontFamily = 'Input Mono, monospace';
});
2019-06-10 10:18:45 +02:00
} else {
2022-12-28 16:05:30 +01:00
code_mirror.style.fontFamily = `${font}, monospace`;
2022-12-28 16:46:17 +01:00
code_mirror.querySelectorAll('pre, textarea').forEach((element) => {
2022-12-28 16:05:30 +01:00
element.style.fontFamily = `${font}, monospace`;
});
2019-06-10 10:18:45 +02:00
}
2022-12-28 16:46:17 +01:00
document.querySelectorAll('#select-font [data-alias]').forEach((element) => {
element.classList.remove('active');
});
2022-12-28 16:46:17 +01:00
document.querySelectorAll(`#select-font [data-alias='${font}']`).forEach((element) => {
element.classList.add('active');
});
2019-06-10 10:18:45 +02:00
2022-12-28 16:05:30 +01:00
document.cookie = `font=${font};max-age=172800`;
2022-12-28 16:46:17 +01:00
};
2017-01-20 17:59:59 +01:00
2017-07-28 23:07:09 +02:00
window.onhashchange = selectFont;
2017-01-20 17:59:59 +01:00
function setSize() {
var size = document.getElementById('size').value;
2017-09-02 21:26:30 +02:00
2022-12-28 16:05:30 +01:00
document.querySelector('.CodeMirror').style.fontSize = `${size}px`;
document.cookie = `size=${size};max-age=172800`;
editor.refresh();
2017-01-20 17:59:59 +01:00
}
function setSpacing() {
var spacing = document.getElementById('spacing').value;
2017-09-02 21:26:30 +02:00
document.querySelector('.CodeMirror').style.lineHeight = spacing;
2022-12-28 16:05:30 +01:00
document.cookie = `spacing=${spacing};max-age=172800`;
editor.refresh();
2017-01-20 17:59:59 +01:00
}
function selectLanguage() {
var lang = document.getElementById('select-language').value;
2017-09-02 21:26:30 +02:00
2019-06-10 10:18:45 +02:00
editor.setOption('mode', lang.toLowerCase());
2022-12-28 16:05:30 +01:00
document.cookie = `language=${lang};max-age=172800`;
2017-01-20 17:59:59 +01:00
}
2019-06-10 20:58:18 +02:00
function setCounter(amount) {
var counter_element = document.querySelector('h1 a:first-child');
2019-06-10 20:58:18 +02:00
if (amount === 1) {
2022-12-28 16:05:30 +01:00
counter_element.innerHTML = `${amount} Programming Font`;
2019-06-10 20:58:18 +02:00
} else {
2022-12-28 16:05:30 +01:00
counter_element.innerHTML = `${amount} Programming Fonts`;
2019-06-10 20:58:18 +02:00
}
}
2017-01-20 17:59:59 +01:00
function applyFilters() {
var count = 0;
2022-12-28 16:46:17 +01:00
Object.keys(filters).forEach((filter) => {
2022-12-28 16:05:30 +01:00
var button = document.querySelector(`button[value="${filter}"]`);
if (!button) {
2020-10-07 22:25:17 +02:00
return;
}
if (filters[filter]) {
button.classList.add('selected');
2022-12-28 16:46:17 +01:00
button.querySelectorAll('svg').forEach((image) => {
2020-10-07 22:25:17 +02:00
image.classList.remove('selected');
});
2022-12-28 16:05:30 +01:00
button.querySelector(`svg[alt="${filters[filter]}"]`).classList.add('selected');
2020-10-07 22:25:17 +02:00
} else {
button.classList.remove('selected');
2022-12-28 16:46:17 +01:00
button.querySelectorAll('svg').forEach((image) => {
2020-10-07 22:25:17 +02:00
image.classList.remove('selected');
});
}
});
2022-12-28 16:46:17 +01:00
document.querySelectorAll('.entry[data-alias]').forEach((element) => {
var data = font_data[element.dataset.alias];
if (
(!filters.style || data.style === filters.style) &&
(!filters.rendering || data.rendering === filters.rendering) &&
2022-12-28 16:05:30 +01:00
(!filters.liga ||
(data.ligatures === false && filters.liga === 'no') ||
(data.ligatures === true && filters.liga === 'yes')) &&
(!filters.zerostyle || data.zerostyle === filters.zerostyle) &&
(filters.author === 'all' || data.author === filters.author) &&
(!filters.name || data.name.toLowerCase().indexOf(filters.name) > -1)
) {
element.classList.remove('filtered-out');
count++;
} else {
element.classList.add('filtered-out');
}
});
setCounter(count);
}
2019-01-22 19:45:16 +01:00
function renderSelectList() {
2022-12-28 16:05:30 +01:00
var icon =
'<svg class="octicon" viewBox="0 0 12 14" version="1.1" width="12" height="14" aria-hidden="true"><path fill-rule="evenodd" d="M11 10h1v3c0 .55-.45 1-1 1H1c-.55 0-1-.45-1-1V3c0-.55.45-1 1-1h3v1H1v10h10v-3zM6 2l2.25 2.25L5 7.5 6.5 9l3.25-3.25L12 8V2H6z"></path></svg>';
var pinIcon =
'<svg class="octicon octicon-pin" viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true"><path fill-rule="evenodd" d="M10 1.2V2l.5 1L6 6H2.2c-.44 0-.67.53-.34.86L5 10l-4 5 5-4 3.14 3.14a.5.5 0 0 0 .86-.34V10l3-4.5 1 .5h.8c.44 0 .67-.53.34-.86L10.86.86a.5.5 0 0 0-.86.34z"></path></svg>';
2019-06-10 10:18:45 +02:00
var favoritesMap = {};
var favorites = [];
var ajax = new XMLHttpRequest();
2019-06-10 10:18:45 +02:00
document.getElementById('select-font').innerHTML = '';
2019-06-10 10:18:45 +02:00
try {
favorites = JSON.parse(localStorage.getItem('favorites')) || [];
2022-12-28 16:05:30 +01:00
favoritesMap = favorites.reduce((acc, alias) => {
2019-06-10 10:18:45 +02:00
acc[alias] = true;
return acc;
}, {});
} catch (err) {
// eslint-disable-next-line no-console
console.error('could not render favorites', err);
}
2022-12-28 16:05:30 +01:00
ajax.onreadystatechange = () => {
var fonts = [];
var authors = [];
if (ajax.readyState === 4 && ajax.status === 200) {
font_data = ajax.response;
2022-12-28 16:46:17 +01:00
Object.keys(font_data).forEach((key) => {
2022-12-28 16:05:30 +01:00
var v = font_data[key];
v.alias = key;
fonts.push(v);
if (authors.indexOf(v.author) < 0) {
authors.push(v.author);
}
});
}
authors.sort();
2022-12-28 16:05:30 +01:00
fonts.sort((a, b) => {
if (favoritesMap[a.alias] && !favoritesMap[b.alias]) {
return -1;
}
if (!favoritesMap[a.alias] && favoritesMap[b.alias]) {
return 1;
}
if (a.name.toLowerCase() < b.name.toLowerCase()) {
return -1;
}
if (a.name.toLowerCase() > b.name.toLowerCase()) {
return 1;
}
2019-06-10 10:18:45 +02:00
return 0;
});
2022-12-28 16:46:17 +01:00
authors.forEach((author) => {
var option = document.createElement('option');
option.innerHTML = author;
document.getElementById('authors-list').querySelector('.other').appendChild(option);
});
2022-12-28 16:46:17 +01:00
fonts.forEach((v) => {
var option = document.createElement('div');
2019-06-10 10:18:45 +02:00
option.classList.add('entry');
if (favoritesMap[v.alias]) {
option.classList.add('pinned');
}
2020-10-14 19:51:46 +02:00
option.setAttribute('data-alias', v.alias);
2022-12-28 16:46:17 +01:00
option.innerHTML = `<a href="#${v.alias}" data-style="${v.style}"><span class="name">${v.name}</span><span class="details">${v.year}${v.author}</span></a><a class="favoritelink" onclick="toggleFavorite(\'${v.alias}\')">${pinIcon}</a><a class="website" href="${v.website}" rel="external"> <span>Website</span>${icon}</a>`;
document.getElementById('select-font').appendChild(option);
2019-06-10 10:18:45 +02:00
});
2019-06-10 10:18:45 +02:00
selectFont();
applyFilters();
};
ajax.responseType = 'json';
ajax.open('GET', 'fonts.json', true);
ajax.send();
2019-01-22 19:45:16 +01:00
}
2019-06-10 10:18:45 +02:00
// eslint-disable-next-line no-unused-vars
2019-01-22 19:45:16 +01:00
function toggleFavorite(alias) {
2019-06-10 10:18:45 +02:00
var favorites = [];
try {
favorites = JSON.parse(localStorage.getItem('favorites')) || [];
if (favorites.indexOf(alias) > -1) {
2022-12-28 16:46:17 +01:00
favorites = favorites.filter((v) => {
2019-06-10 10:18:45 +02:00
return v !== alias;
});
} else {
favorites.push(alias);
}
localStorage.setItem('favorites', JSON.stringify(Array.from(new Set(favorites))));
} catch (err) {
// eslint-disable-next-line no-console
console.error('could not save favorite', err);
}
renderSelectList();
return false;
2019-01-22 19:45:16 +01:00
}
function walk(direction) {
2019-10-21 20:53:49 +02:00
var activeEntry = document.querySelector('.entry.active');
var target = null;
var next = direction === 'up' ? activeEntry.previousElementSibling : activeEntry.nextElementSibling;
while (target === null) {
if (next) {
if (next.matches('.entry:not(.filtered-out)')) {
target = next;
} else {
next = direction === 'up' ? next.previousElementSibling : next.nextElementSibling;
}
} else {
target = false;
}
2019-06-10 10:18:45 +02:00
}
2019-01-23 21:08:51 +01:00
function isVisible(el) {
var offset = document.getElementById('filters').getBoundingClientRect().height;
var container = document.querySelector('section.select-list').getBoundingClientRect().height;
if (direction === 'up') {
if (el.getBoundingClientRect().top < offset) {
return false;
}
} else {
if (el.getBoundingClientRect().bottom > offset + container) {
return false;
}
}
return true;
}
if (target) {
target.querySelector('a').click();
2022-12-28 16:05:30 +01:00
if (!isVisible(target)) {
target.scrollIntoView();
}
2019-06-10 10:18:45 +02:00
}
2019-01-23 21:08:51 +01:00
}
function increaseFontSize() {
2019-06-10 10:18:45 +02:00
var sizeEl = document.getElementById('size');
sizeEl.value = Number(sizeEl.value) + 1;
sizeEl.onchange();
}
function decreaseFontSize() {
2019-06-10 10:18:45 +02:00
var sizeEl = document.getElementById('size');
sizeEl.value = Number(sizeEl.value) - 1;
sizeEl.onchange();
}
2020-10-07 22:25:17 +02:00
function toggleFilter(filter) {
// cycle through the possible values for each filter
// and set the filters[filter] value,
// or at the end of the cycle set it to false
var options = {
2022-12-28 16:05:30 +01:00
style: [false, 'sans', 'serif'],
rendering: [false, 'vector', 'bitmap'],
liga: [false, 'yes', 'no'],
zerostyle: [false, 'slashed', 'dotted', 'empty'],
2020-10-07 22:25:17 +02:00
};
var current_index = options[filter].indexOf(filters[filter]);
var next = current_index + 1;
if (next < options[filter].length) {
filters[filter] = options[filter][next];
} else {
filters[filter] = options[filter][0];
2019-06-10 14:58:21 +02:00
}
applyFilters();
}
2022-12-28 16:05:30 +01:00
window.addEventListener('DOMContentLoaded', () => {
2019-06-10 10:18:45 +02:00
var cookieValueSpacing = document.cookie.replace(/(?:(?:^|.*;\s*)spacing\s*=\s*([^;]*).*$)|^.*$/, '$1');
var cookieValueSize = document.cookie.replace(/(?:(?:^|.*;\s*)size\s*=\s*([^;]*).*$)|^.*$/, '$1');
var cookieValueTheme = document.cookie.replace(/(?:(?:^|.*;\s*)theme\s*=\s*([^;]*).*$)|^.*$/, '$1');
var cookieValueLanguage = document.cookie.replace(/(?:(?:^|.*;\s*)language\s*=\s*([^;]*).*$)|^.*$/, '$1');
if (cookieValueSpacing !== '') {
document.getElementById('spacing').value = cookieValueSpacing;
2019-06-10 10:18:45 +02:00
}
if (cookieValueSize !== '') {
document.getElementById('size').value = cookieValueSize;
2019-06-10 10:18:45 +02:00
}
if (cookieValueTheme !== '') {
document.getElementById('select-theme').value = cookieValueTheme;
2019-06-10 10:18:45 +02:00
}
if (cookieValueLanguage !== '') {
document.getElementById('select-language').value = cookieValueLanguage;
2019-06-10 10:18:45 +02:00
}
2019-06-10 20:58:18 +02:00
renderSelectList();
2019-06-10 10:18:45 +02:00
selectTheme();
setSize();
setSpacing();
selectLanguage();
function walkThemes(direction) {
var select = document.getElementById('select-theme');
var current = select.selectedOptions[0];
var next;
if (current) {
next = direction === 'up' ? current.previousElementSibling : current.nextElementSibling;
}
if (next) {
select.value = next.value;
}
2019-06-10 10:18:45 +02:00
selectTheme();
}
2019-06-10 10:18:45 +02:00
2022-12-28 16:05:30 +01:00
document.getElementById('theme-next').onclick = () => {
walkThemes('down');
};
2019-06-10 10:18:45 +02:00
2022-12-28 16:05:30 +01:00
document.getElementById('theme-previous').onclick = () => {
walkThemes('up');
};
2022-12-28 16:05:30 +01:00
document
.getElementById('filters')
.querySelectorAll('button')
2022-12-28 16:46:17 +01:00
.forEach((button) => {
2022-12-28 16:05:30 +01:00
button.onclick = (event) => {
event.preventDefault();
event.stopPropagation();
toggleFilter(button.value);
};
});
2021-04-22 20:34:59 +02:00
2022-12-28 16:05:30 +01:00
document.getElementById('authors-list').onchange = (event) => {
filters.author = event.target.value;
applyFilters();
};
2021-04-22 20:34:59 +02:00
document.getElementById('name-search').onkeyup = (event) => {
filters.name = event.target.value.toLowerCase();
2019-06-10 20:58:18 +02:00
applyFilters();
};
2019-06-10 14:58:21 +02:00
2022-12-28 16:05:30 +01:00
document.querySelector('.select-list').onkeyup = (event) => {
2021-04-22 20:34:59 +02:00
if (event.ctrlKey || event.altKey || event.metaKey || event.shiftKey) {
return;
2019-06-10 10:18:45 +02:00
}
2021-04-22 20:34:59 +02:00
if (event.key === 'ArrowUp') {
event.preventDefault();
event.stopPropagation();
walk('up');
} else if (event.key === 'ArrowDown') {
event.preventDefault();
event.stopPropagation();
walk('down');
}
};
2019-06-10 10:18:45 +02:00
2022-12-28 16:05:30 +01:00
document.body.addEventListener('keydown', (event) => {
2019-06-10 10:18:45 +02:00
if (event.ctrlKey || event.metaKey) {
if (event.key === '-') {
event.preventDefault();
event.stopPropagation();
decreaseFontSize();
} else if (event.key === '=') {
event.preventDefault();
event.stopPropagation();
increaseFontSize();
}
}
});
2017-01-20 17:59:59 +01:00
});