1
0
Fork 0
mirror of https://github.com/Findus23/lw1.at.git synced 2024-09-10 05:13:46 +02:00

many more changes

This commit is contained in:
Lukas Winkler 2022-04-24 22:32:55 +02:00
parent 2bab4d71af
commit 0bcea6fad5
Signed by: lukas
GPG key ID: 54DE4D798D244853
45 changed files with 923 additions and 646 deletions

View file

@ -1,4 +1,6 @@
all: extract update compile
babel: extract update compile
publish: rmpublic esbuild build upload
extract:
@ -10,5 +12,15 @@ update:
compile:
pybabel compile -d translations --statistics
publish:
rsync -aPz public/* lw1.at:/var/www/beta.lw1.at/
upload:
rsync -aPz --delete-after public/* lw1.at:/var/www/beta.lw1.at/
esbuild:
npm run build
build:
/home/lukas/.virtualenvs/lw1/bin/python lw1/main.py
rmpublic:
rm -r public/
mkdir public/

BIN
assets/background.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

View file

@ -4,9 +4,7 @@ export function initFeedback() {
if (feedbackButton) {
feedbackButton.addEventListener("click", ev => {
const id = feedbackButton.dataset.id
// @ts-ignore
if (typeof window._paq != "undefined") {
// @ts-ignore
window._paq.push(['trackEvent', 'Feedback', 'readmore', id]);
} else {
console.info("Feedback not sent as Matomo isn't loaded");

View file

@ -1,6 +1,7 @@
import {decode} from "blurhash";
import {initFeedback} from "./feedback";
import {initSearch} from "./search";
import {initMatomo} from "./matomo";
const width = 32;
const height = width / 2;
@ -39,3 +40,4 @@ if (canvasList.length > 5) {
initFeedback()
initSearch()
initMatomo()

37
assets/matomo.ts Normal file
View file

@ -0,0 +1,37 @@
import {isDebug} from "./utils"
declare global {
interface Window {
_paq: paq
__PRERENDER_INJECTED: boolean
}
}
declare type paq = [[string, unknown, unknown, unknown] | [string, unknown, unknown] | [string, unknown] | [string]]
export function initMatomo(): void {
if (isDebug) {
return
}
const _paq = window._paq || [] as unknown as paq
_paq.push(['setRequestMethod', 'POST'])
_paq.push(['enableHeartBeatTimer'])
_paq.push(["setDoNotTrack", true])
_paq.push(['disableCookies'])
const matomoURL = "https://matomo.lw1.at/statistics"
_paq.push(['setTrackerUrl', matomoURL + ".php"])
_paq.push(['setSiteId', 14])
const script = document.createElement('script')
const firstScript = document.getElementsByTagName('script')[0]
script.type = 'text/javascript'
script.async = true
script.defer = true
script.src = matomoURL + ".js"
if (firstScript.parentNode) {
firstScript.parentNode.insertBefore(script, firstScript)
}
window._paq = _paq
}

View file

@ -17,7 +17,7 @@ export function redirect(): void {
return
}
l.replace(l.pathname + defaultLanguage() + "/")
l.replace("/" + defaultLanguage() + l.pathname)
}
// redirect()
redirect()

5
assets/scss/README.md Normal file
View file

@ -0,0 +1,5 @@
partially based on milligram.css
The MIT License (MIT)
Copyright © 2016 CJ Patoilo <cjpatoilo@gmail.com>

18
assets/scss/_base.scss Normal file
View file

@ -0,0 +1,18 @@
@import "variables";
body {
background-color: white;
background-repeat: repeat;
background-image: url("../background.png");
image-rendering: pixelated;
image-rendering: -moz-crisp-edges;
image-rendering: crisp-edges;
background-size: 512px*2;
background-attachment: fixed;
margin: 0;
> * {
image-rendering: initial;
}
}

19
assets/scss/_boxes.scss Normal file
View file

@ -0,0 +1,19 @@
@import "variables";
.note {
background-color: $consoleBackground;
color: $consoleOrange;
font-family: $consoleFont;
text-align: center;
/*background-color: #fdbc4b;*/
padding: 15px;
margin-bottom: 15px;
a {
color: $consoleBlue;
&:hover, &:focus, &:active {
text-decoration: underline;
}
}
}

33
assets/scss/_btn.scss Normal file
View file

@ -0,0 +1,33 @@
@import "variables";
.btn, button, .tag, #filterwrapper > a,
input[type='button'],
input[type='reset'],
input[type='submit'] {
background: $consoleBackground;
font-family: $consoleFont;
padding: .5rem;
color: $consoleText;
margin: auto auto 1.5rem;
line-height: initial;
border: none;
&:focus, &:hover, &:active {
outline: $consoleBackground solid 1px;
background-color: white;
color: $textColor;
}
&:active {
color: $consoleBlue;
}
}
button,
input[type='button'],
input[type='reset'],
input[type='submit'] {
cursor: pointer;
}

View file

@ -1,21 +1,22 @@
.contact {
display: flex;
justify-content: space-around;
flex-wrap: wrap;
svg {
width: 45px;
height: 45px;
color: $consoleBackground;
transition: color .2s;
padding: 5px;
width: 45px;
height: 45px;
color: $consoleBackground;
transition: color .2s;
padding: 5px;
&.keybase .line {
fill: currentColor;
}
&.keybase .line {
fill: currentColor;
}
}
a:hover svg {
color: $consoleFocus;
outline: 1px solid $consoleFocus;
outline: 1px solid $consoleFocus;
}
}

View file

@ -0,0 +1,16 @@
@import "variables";
.container {
max-width: 1000px;
position: relative;
width: 100%;
background-color: white;
border: 1px solid $consoleBackground;
text-align: center;
margin: 20px auto 0;
@media (max-width: 1000px) {
border: none;
margin-top: 0;
}
}

View file

@ -0,0 +1,33 @@
.topbuttons {
@media screen and (max-width: 480px) {
display: flex;
justify-content: center;
}
}
.languageSelector, .backButton {
position: absolute;
top: 0;
z-index: 10000;
margin: 0;
@media screen and (max-width: 480px) {
position: initial;
display: block;
width: 50%;
border: 1px solid white;
}
}
.languageSelector {
right: 0;
}
.backButton {
left: 0;
.arrow {
font-weight: bold;
}
}

View file

@ -0,0 +1,85 @@
/**
* Fork of https://github.com/chinchang/hint.css/blob/master/src/hint-position.scss
* remove all but .hint--bottom
* MIT License
* Copyright (c) 2021 Kushagra Gour
*/
@mixin vertical-positioned-tooltip($propertyY, $transitionDirection, $xDirection:0) {
&:before {
// bring arrow inside so that it animates to normal position when shown.
// HACK: +1px to avoid the 1 px white space between arrow and body during transition.
margin-#{$propertyY}: -2 * $hintArrowBorderWidth + 1px;
}
&:before, &:after {
#{$propertyY}: 100%;
left: 50%; // get top-left corner in center
}
&:before {
left: calc(50% - #{$hintArrowBorderWidth}); // get arrow center aligned with content
}
$translateX: -50%;
@if $xDirection == -1 {
$translateX: -100%;
} @else if $xDirection == 1 {
$translateX: 0;
}
&:after {
@include vendor('transform', translateX($translateX));
}
&:after {
@if $xDirection != 0 {
// bring back the tooltip by some offset so that arrow doesn't stick at end
margin-left: -$xDirection * $hintArrowOffsetX;
}
}
&:hover {
@include set-margin('translateY', $transitionDirection, $translateX);
}
}
@mixin horizontal-positioned-tooltip($propertyX, $transitionDirection) {
&:before {
// bring arrow inside so that it animates to normal position when shown
// HACK: +1px to avoid the 1 px white space between arrow and body during transition.
margin-#{$propertyX}: -2 * $hintArrowBorderWidth + 1px;
// bring back to center vertically
margin-bottom: -1 * $hintArrowBorderWidth;
}
&:after {
// bring back to center
margin-bottom: -1 * floor($hintTooltipHeight / 2);
}
&:before, &:after {
#{$propertyX}: 100%;
bottom: 50%;
}
&:hover {
@include set-margin('translateX', $transitionDirection);
}
}
/**
* set default color for tooltip arrows
*/
.hint--bottom:before {
border-bottom-color: $hintDefaultColor;
}
/**
* bottom tooltip
*/
.#{$hintPrefix}bottom {
@include vertical-positioned-tooltip('top', 1);
}

View file

@ -1,10 +0,0 @@
canvas {
position: absolute;
top: 0;
left: 0;
width: 100%;
}
img.background {
background: #f7f7f7 !important;
}

3
assets/scss/_images.scss Normal file
View file

@ -0,0 +1,3 @@
img {
max-width: 100%;
}

View file

@ -1,6 +1,8 @@
@import "variables";
.intro {
padding-top: 2rem;
}
.introduction {
text-align: left;
@ -14,18 +16,18 @@ h1 {
cursor: initial !important;
div {
font-size: 24px;
line-height: 28px;
font-size: 24px;
line-height: 28px;
}
}
@media screen and (max-width: 480px) {
h1 {
text-align: left;
padding: 0 10px;
text-align: left;
padding: 0 10px;
}
.languageSelector {
right: 10px;
right: 10px;
}
}

23
assets/scss/_others.scss Normal file
View file

@ -0,0 +1,23 @@
li {
margin-bottom: 0.3rem;
}
a svg {
fill: initial;
}
.toImprint {
display: inline-block;
}
[aria-label][data-balloon-pos] {
&::after {
font-family: unset !important;
background: $consoleBackground !important;
}
}

View file

@ -1,12 +1,16 @@
@import "variables";
@import "../node_modules/milligram/src/Color";
@import "../node_modules/milligram/src/Utility";
#blockwrapper {
margin: 20px;
display: flex;
flex-wrap: wrap;
justify-content: space-around;
}
.card {
color: $consoleText;
background: $consoleBackground;
transition: color .2s;
border-radius: $borderRadius;
z-index: 3;
position: relative;
margin-bottom: 20px;
@ -14,19 +18,6 @@
max-width: 300px;
cursor: pointer;
&:before {
border-radius: $borderRadius;
content: "";
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
z-index: -3;
/*@include shadow(2);*/
/*transition: box-shadow 0.2s;*/
}
&:hover, &:focus {
color: $consoleFocus;
@ -36,16 +27,9 @@
}
}
.imgwrapper {
background: $consoleBackground
}
img {
display: block;
transition: filter .2s;
border-top-left-radius: $borderRadius;
border-top-right-radius: $borderRadius;
height: auto;
z-index: 5;
position: relative;
@ -53,6 +37,7 @@
.imgwrapper {
height: 150px;
width: 300px;
}
.textwrapper {
@ -69,47 +54,26 @@
}
.tag {
font-family: $consoleFont;
display: inline-block;
padding: .2em .6em .3em;
font-size: 70%;
font-weight: bold;
line-height: 1;
border: 1px $consoleText solid;
color: $consoleText;
background-color: $consoleBackground;
text-align: center;
white-space: nowrap;
vertical-align: baseline;
border-radius: $borderRadius;
margin-left: 2px;
margin-right: 2px;
transition: color .2s, border-color .2s;
margin: 0 2px;
}
button {
color: white;
.tag, #filterwrapper > a {
font-size: 1.3rem;
font-weight: bold;
line-height: 1;
padding: .3em .6em .3em;
&.colored {
color: white;
}
&.active {
/*color: darken($color-secondary, 10%);
border-color: darken($color-secondary, 10%);
background-color: darken(white, 10%) !important;*/
color: $consoleFocus;
}
svg {
vertical-align: sub;
margin-right: 4px;
}
}
#filterwrapper, #searchwrapper, .contact {
margin: 0 2rem;
}
#filterwrapper {
font-family: monospace, monospace;
margin-bottom: 2rem;
display: flex;
flex-wrap: wrap;
@ -118,21 +82,16 @@ background-color: darken(white, 10%) !important;*/
a {
display: block;
color: $consoleBackground;
//text-transform: initial;
font-family: $consoleFont;
font-size: 1.2rem;
padding: .2em .6em;
//line-height: 10px;
//height: 25px;
font-weight: 700;
letter-spacing: .1rem;
background: white;
&.active {
color: $consoleWhite;
color: $consoleWhite !important;
outline: none !important;
background-color: $consoleBackground;
}
&:focus, &:active, &:hover {
color: $consoleBackground;
outline: 1px solid $consoleBackground;
}
}
@ -145,38 +104,40 @@ background-color: darken(white, 10%) !important;*/
font-family: $consoleFont;
-webkit-appearance: none;
-moz-appearance: none;
padding: 4px;
border: 1px solid #bbb; /* Here */
border: none;
padding: .5rem;
border: 1px solid $consoleBackground;
border-radius: $borderRadius;
margin-bottom: 0;
background-color: $consoleBackground;
color: $consoleText;
background-color: white;
color: $consoleBackground;
max-width: 300px;
width: 100%;
transition: border-color .2s, color .2s;
&:focus {
border-color: $color-primary;
border-color: $consoleBlue;
outline: none;
&::placeholder {
color: $consoleBlue;
opacity: .5;
}
}
&::placeholder {
color: $consoleText;
color: $consoleBackground;
transition: color .2s;
opacity: 1;
}
&::selection {
background: $consoleText;
color: $consoleBackground;
background: $consoleBackground;
color: $consoleWhite;
}
}
}
#noresults {
font-family: $consoleFont;
font-size: 20px;
font-weight: bold;
}
.gpg {
display: block;

View file

@ -1,52 +1,9 @@
.closeButton {
position: absolute;
font-size: 22px;
line-height: 22px;
top: 10px;
right: 10px;
padding: 10px;
cursor: pointer;
transition: color 0.2s;
color: $color-primary;
z-index: 2000;
@import "variables";
&:hover {
color: darkgrey;
}
}
.modal-mask {
background-color: rgba(0, 0, 0, 0.5);
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
overflow: auto;
-webkit-overflow-scrolling: touch;
padding: 0 20px;
@media (max-width: 40.0rem) {
padding: 0 10px;
}
z-index: 1000;
transition: opacity .3s ease;
}
//.modal-container {
// position: relative;
// max-width: 1000px;
// /*
// @media (max-width: 40.0rem) {
// width: 100%;
// padding: 20px;
// }
//*/
// margin: 50px auto 0;
// background-color: #fff;
// border-radius: $borderRadius;
.imagewrapper {
padding-bottom: 50%;
//padding-bottom: 50%;
position: relative;
background: white;
&.separator {
@ -56,31 +13,20 @@
img {
display: block;
width: 100%;
border-top-left-radius: $borderRadius;
border-top-right-radius: $borderRadius;
height: auto;
position: absolute;
left: 0;
top: 0;
position: relative;
z-index: 1100;
}
}
//box-shadow: 0 3px 6px rgba(0, 0, 0, 0.16), 0 3px 6px rgba(0, 0, 0, 0.23);
//
//&:focus {
//outline: none;
//}
//}
.modal-header h3 {
margin-top: 0;
color: #42b983;
canvas {
position: absolute;
top: 0;
left: 0;
width: 100%;
}
h1 {
margin: 0 5px;
}
.modal-body {
padding: 20px 30px;
@ -88,31 +34,6 @@ h1 {
text-align: left;
}
.modal-default-button {
float: right;
}
/*
* The following styles are auto-applied to elements with
* transition="modal" when their visibility is toggled
* by Vue.js.
*
* You can easily play with the modal transition by editing
* these styles.
*/
.modal-enter {
opacity: 0;
}
.modal-leave-active {
opacity: 0;
}
/*.modal-enter .modal-container,*/
/*.modal-leave-active .modal-container {*/
/*transform: scale(0.8);*/
/*}*/
.modal-linkbar {
display: flex;
@ -140,12 +61,8 @@ h1 {
display: block;
&:hover {
svg {
color: $color-primary;
}
.try-it-out {
color: $color-primary;
svg, .try-it-out {
color: $linkColor;
}
}
@ -162,10 +79,6 @@ h1 {
position: relative;
top: 20px;
}
span {
}
}
}
}
@ -179,12 +92,7 @@ iframe {
margin-bottom: 15px;
}
.note {
background-color: $consoleBackground;
color: $consoleOrange;
font-family: $consoleFont;
text-align: center;
/*background-color: #fdbc4b;*/
padding: 15px;
margin-bottom: 15px;
#readmore {
display: block;
}

19
assets/scss/_spacing.scss Normal file
View file

@ -0,0 +1,19 @@
.button,
button,
input,
textarea,
select,
fieldset {
margin-bottom: 1rem
}
pre,
blockquote,
dl,
figure,
table,
p,
ul,
ol {
margin-bottom: 1.5rem
}

View file

@ -0,0 +1,116 @@
@import "variables";
html {
font-size: 62.5%;
}
body {
color: $textColor;
font-size: 1.8em;
font-weight: normal;
font-family: $fontFamily;
line-height: 1.6;
}
p {
margin-top: 0;
hyphens: auto;
}
h1,
h2,
h3,
h4,
h5,
h6 {
font-weight: 300;
//letter-spacing: -.1rem;
margin-bottom: 2.0rem;
margin-top: 0;
}
h1 {
font-size: 4.0rem;
line-height: 1.2;
}
h2 {
font-size: 3.6rem;
line-height: 1.25;
}
h3 {
font-size: 3.0rem;
line-height: 1.3;
}
h4 {
font-size: 2.4rem;
letter-spacing: -.08rem;
line-height: 1.35;
}
h5 {
font-size: 1.8rem;
letter-spacing: -.05rem;
line-height: 1.5;
}
h6 {
font-size: 1.6rem;
letter-spacing: 0;
line-height: 1.4;
}
.texttt {
font-family: $consoleFont;
}
pre {
white-space: pre-wrap;
}
blockquote {
border-left: .3rem solid $gray;
margin-left: 0;
margin-right: 0;
padding: 1rem 1.5rem;
*:last-child {
margin-bottom: 0
}
}
a {
color: $linkColor;
text-decoration: none;
}
a, .btn, button, .tag {
transition: background-color 0.2s ease-in-out, color 0.2s, border-color 0.2s
}
table {
border-spacing: 0;
width: 100%;
}
th,
td {
border-bottom: .1rem solid #e1e1e1;
padding: 1.2rem 1.5rem;
text-align: left;
&:first-child {
padding-left: 0
}
&:last-child {
padding-right: 0
}
}

View file

@ -1,189 +1,30 @@
@import "variables";
@import "hint.css/src/hint.base";
@import "hint.css/src/hint-variables";
@import "hint.css/src/hint-mixins";
@import "hint.css/src/hint-core";
@import "hint-position";
@import "../../node_modules/modern-normalize/modern-normalize";
@import "variables";
@import "../../node_modules/milligram/src/Color";
@import "../../node_modules/milligram/src/Vars";
@import "../../node_modules/milligram/src/Base";
@import "../../node_modules/milligram/src/Blockquote";
@import "../../node_modules/milligram/src/Blocks";
//@import "../../node_modules/milligram/src/Button";
//@import "../../node_modules/milligram/src/Code";
@import "../../node_modules/milligram/src/Divider";
@import "../../node_modules/milligram/src/Form";
@import "../../node_modules/milligram/src/Grid";
@import "../../node_modules/milligram/src/Images";
@import "../../node_modules/milligram/src/Label";
@import "../../node_modules/milligram/src/Link";
@import "../../node_modules/milligram/src/List";
@import "../../node_modules/milligram/src/Navbar";
@import "../../node_modules/milligram/src/Spacing";
@import "../../node_modules/milligram/src/Table";
@import "../../node_modules/milligram/src/Typography";
@import "../../node_modules/milligram/src/Utility";
@import "btn";
@import "base";
@import "typography";
@import "boxes";
@import "spacing";
@import "images";
@import "container";
@import "cornerbuttons";
@import "post";
@import "intro";
@import "contact";
@import "overview";
@import "image";
@import "code";
@import "highlight";
@import "license";
@import "imprint";
@import "langpick";
pre {
white-space: pre-wrap;
}
body {
background-color: white;
background-repeat: repeat;
background-image: url("");
image-rendering: pixelated;
image-rendering: -moz-crisp-edges;
image-rendering: crisp-edges;
background-size: 512px*2;
background-attachment: fixed;
font-family: -apple-system, "Helvetica Neue Light", "HelveticaNeue", "Helvetica Neue", "Roboto", "Liberation Sans", Arial, sans-serif;
font-weight: normal;
color: #212121;
> * {
image-rendering: initial;
}
}
html {
scrollbar-color: $consoleBackground white;
scrollbar-width: auto;
//scrollbar-gutter:stable
}
li {
margin-bottom: 0.3rem;
}
a svg {
fill: initial;
}
.container {
background-color: white;
border: 1px solid $consoleBackground;
@media (max-width: 1120px) {
border: none;
}
max-width: 1000px;
padding-top: 2rem;
}
#app {
text-align: center;
margin-top: 20px;
}
#blockwrapper {
margin: 20px;
display: flex;
flex-wrap: wrap;
justify-content: space-around;
}
.languageSelector a, .backButton a, .toImprint {
&:hover, &:focus {
color: $consoleBlue;
}
}
.languageSelector, .backButton {
position: absolute;
background: $consoleBackground;
font-family: $consoleFont;
a {
padding: 5px;
color: $consoleText;
}
@media screen and (max-width: 480px) {
right: 10px;
}
}
.languageSelector, .backButton {
top: 0;
z-index: 10000;
}
.languageSelector {
right: 0;
}
.backButton {
left: 0;
.arrow {
font-weight: bold;
}
}
.toImprint {
display: inline-block;
background: $consoleBackground;
font-family: $consoleFont;
padding: 5px;
color: $consoleText;
margin: auto auto 1.5rem;
}
.texttt {
font-family: $consoleFont;
}
.button,
button,
input[type='button'],
input[type='reset'],
input[type='submit'] {
background-color: $consoleBackground;
font-family: $consoleFont;
cursor: pointer;
display: inline-block;;
font-size: 1.1rem;
font-weight: 700;
//height: 3.8rem;
letter-spacing: .1rem;
line-height: 3.8rem;
padding: 0 3.0rem;
text-align: center;
text-decoration: none;
//white-space: nowrap;
border: none;
&:focus, &:hover, &:active {
outline: $consoleBackground solid 1px;
background-color: transparent;
color: $consoleBackground;
}
&:active {
background-color: #eee;
}
}
[aria-label][data-balloon-pos] {
&::after {
font-family: unset !important;
background: $consoleBackground !important;
}
}
@import "others";

View file

@ -2,6 +2,10 @@ $color-primary: #303f9f;
$borderRadius: 0px;
$imageheight: 150px;
$textColor: #212121;
$linkColor: #303f9f;
$consoleBackground: #31363B;
$consoleWhite: #EFF0F1;
$consoleOrange: #FDBC4B;
@ -9,10 +13,14 @@ $consoleBlue: #3DAEE9;
$consoleGreen: #16A085;
$consoleRed: #963830;
$gray: #cacacc;
$consoleFocus: $consoleBlue;
$consoleText: $consoleWhite;
$consoleFont: "Fira Code Retina", monospace, monospace;
$fontFamily: -apple-system, "Helvetica Neue Light", "HelveticaNeue", "Helvetica Neue", "Liberation Sans", "Roboto", Arial, sans-serif;
@mixin dark() {
@media (prefers-color-scheme: dark) {

2
assets/utils.ts Normal file
View file

@ -0,0 +1,2 @@
export const isDebug = document.body.dataset.debug == "True"

View file

@ -14,7 +14,8 @@ const commonOption: BuildOptions = {
loader: {
".ttf": "file",
".woff": "file",
".woff2": "file"
".woff2": "file",
".png": "base64"
},
entryNames: "[name]"
}
@ -23,7 +24,7 @@ const commonOption: BuildOptions = {
switch (process.argv[2]) {
case "build":
commonOption.metafile = true
commonOption.entryNames = "[dir]/[name]-[hash]"
commonOption.entryNames = "[name]-[hash]"
build(commonOption).then(result => {
fs.writeFileSync(commonOption.outdir + "/meta.json", JSON.stringify(result.metafile))
})

View file

@ -1,4 +1,4 @@
id: masterarbeit
#id: masterarbeit
tags:
- python
- latex

View file

@ -7,6 +7,7 @@ from jinja2 import Environment, FileSystemLoader, select_autoescape, StrictUndef
from lw1.paths import templates_dir, cache_dir
from lw1.post import Post
from lw1.settings import FALLBACK_LANGUAGES
from lw1.sitemap import Sitemap
from lw1.tag import Tag
from lw1.types import Language
from lw1.writer import Writer
@ -26,10 +27,11 @@ class Generator:
type = ""
def __init__(self, translations: Translations, context: Dict, posts: List[Post], tags: Dict[str, Tag],
lang: Language, ):
lang: Language, sitemap: Sitemap):
self.posts = posts
self.tags = tags
self.lang = lang
self.sitemap = sitemap
self.context = context.copy()
self.context["js_entry_point"] = self.js_entry_point
self.context["type"] = self.type
@ -45,21 +47,17 @@ class PostsGenerator(Generator):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
def url(self, post: Post) -> Path:
print(post.title)
return Path("") / self.lang / post.slug
def generate(self, writer: Writer) -> None:
for post in self.posts:
if post.lang != self.lang:
continue
for image in {post.content_img, post.thumbnail_img}:
writer.writeImage((Path("") / "img" / post.id), image)
writer.writeImage((Path("") / "img" / post.slug), image)
self.context["post"] = post
self.context["otherlang_url"] = post.other_lang_post.absolute_url
html = self.template.render(**self.context)
url = self.url(post)
writer.write(url, html)
writer.write(post.absolute_url, html)
self.sitemap.add_post(post)
class HomepageGenerator(Generator):
@ -75,16 +73,19 @@ class HomepageGenerator(Generator):
self.context["current"] = "all"
self.context["posts"] = [p for p in self.posts if p.lang == self.lang]
self.context["otherlang_url"] = Path("/") / other_lang
self.context["otherlang"] = other_lang
url = Path("/") / self.lang
self.context["url"] = url
self.sitemap.add_page(self.context)
html = self.template.render(**self.context)
url = Path("") / self.lang
writer.write(url, html)
for tag in self.tags.values():
self.context["current"] = tag.id
if tag.hidden:
continue
self.context["posts"] = [p for p in self.posts if p.lang == self.lang and tag.id in p.tags]
url = Path("") / self.lang / "t" / tag.slug_in_lang(self.lang)
url = Path("/") / self.lang / "t" / tag.slug_in_lang(self.lang)
self.context["otherlang_url"] = Path("/") / other_lang / "t" / tag.slug_in_lang(other_lang)
html = self.template.render(**self.context)
@ -103,10 +104,13 @@ class ImprintGenerator(Generator):
super().__init__(*args, **kwargs)
def generate(self, writer: Writer) -> None:
fb_lang = FALLBACK_LANGUAGES[self.lang]
self.context["otherlang_url"] = Path("/") / fb_lang / self.slugs[fb_lang]
other_lang = FALLBACK_LANGUAGES[self.lang]
url = Path("/") / self.lang / self.slugs[self.lang]
self.context["otherlang_url"] = Path("/") / other_lang / self.slugs[other_lang]
self.context["otherlang"] = other_lang
self.context["url"] = url
self.sitemap.add_page(self.context)
html = self.template.render(**self.context)
url = Path("") / self.lang / self.slugs[self.lang]
writer.write(url, html)
@ -119,6 +123,9 @@ class LangRedirectGenerator(Generator):
def generate(self, writer: Writer) -> None:
html = self.template.render(**self.context)
root = Path("")
for url in [root, root / "i"]:
root = Path("/")
paths = {root, root / "i"}
for post in self.posts:
paths.add(root / post.slug)
for url in paths:
writer.write(url, html)

View file

@ -4,11 +4,14 @@ from babel.support import Translations
from lw1.generators import PostsGenerator, HomepageGenerator, ImprintGenerator, LangRedirectGenerator
from lw1.loader import PostLoader, TagsLoader, AssetsLoader
from lw1.paths import output_dir
from lw1.settings import LANGUAGES
from lw1.sitemap import Sitemap
from lw1.writer import Writer
def main(debug=False):
sitemap = Sitemap()
start = perf_counter_ns()
posts = PostLoader.load_posts()
tags = TagsLoader.load_tags()
@ -25,8 +28,14 @@ def main(debug=False):
LangRedirectGenerator
]
for cls in generators:
g = cls(posts=posts, tags=tags, lang=lang, context=context, translations=translations)
g = cls(
posts=posts, tags=tags,
lang=lang, context=context,
translations=translations,
sitemap=sitemap
)
g.generate(writer)
sitemap.dump(output_dir / "sitemap.xml")
end = perf_counter_ns()
print(f"{(end - start) / 1000 / 1000:.2f} ms")

View file

@ -2,13 +2,14 @@ from __future__ import annotations
from dataclasses import dataclass
from functools import cached_property
from pathlib import Path
from typing import Dict, List
from slugify import slugify
from lw1.file import Image
from lw1.settings import DOMAIN
from lw1.tomarkdown import markdown2html
from lw1.types import Language
from lw1.utils import custom_slugify
@dataclass
@ -20,20 +21,21 @@ class License:
@dataclass
class iFrame:
url: str
allowfullscreen: bool
sandbox: str
color: str
sandbox: str = None
allowfullscreen: bool = False
@dataclass
class Post:
id: str
title: str
tags: List[str]
date: str
markdown: str
content_img: Image
thumbnail_img: Image