mirror of
https://github.com/Steffo99/bluelib.git
synced 2024-12-23 03:54:21 +00:00
First commit (0.8.1)
This commit is contained in:
commit
f034bf23ba
34 changed files with 13171 additions and 0 deletions
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
node_modules
|
||||||
|
/*.log
|
||||||
|
.idea/
|
||||||
|
size-plugin.json
|
12354
package-lock.json
generated
Normal file
12354
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
32
package.json
Normal file
32
package.json
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
{
|
||||||
|
"private": false,
|
||||||
|
"name": "bluelib",
|
||||||
|
"version": "0.8.1",
|
||||||
|
"license": "AGPL-3.0-or-later",
|
||||||
|
"main": "src/index.js",
|
||||||
|
"scripts": {
|
||||||
|
"start": "preact watch",
|
||||||
|
"build": "preact build --no-prerender --dest docs && git add docs",
|
||||||
|
"ghpages": "npm run -s build && git add . && cross-env-shell git commit -m \"$npm_package_version\" && git push && cross-env-shell hub release create -m \"$npm_package_version\" \"$npm_package_version\" && cross-env-shell sentry-cli releases set-commits \"$npm_package_version\" --auto && cross-env-shell sentry-cli releases deploys \"$npm_package_version\" new --env production -n \"gh-pages\""
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"cross-env": "^7.0.2",
|
||||||
|
"preact-cli": "^3.0.0-rc.14"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@fortawesome/fontawesome-svg-core": "^1.2.28",
|
||||||
|
"@fortawesome/free-brands-svg-icons": "^5.13.0",
|
||||||
|
"@fortawesome/free-regular-svg-icons": "^5.13.0",
|
||||||
|
"@fortawesome/free-solid-svg-icons": "^5.13.0",
|
||||||
|
"@fortawesome/react-fontawesome": "^0.1.9",
|
||||||
|
"css-loader": "^3.5.3",
|
||||||
|
"file-loader": "^5.0.2",
|
||||||
|
"less": "^3.11.1",
|
||||||
|
"less-loader": "^5.0.0",
|
||||||
|
"preact": "^10.4.4",
|
||||||
|
"preact-render-to-string": "^5.1.8",
|
||||||
|
"react-syntax-highlighter": "^12.2.1",
|
||||||
|
"showdown": "^1.9.1",
|
||||||
|
"style-loader": "^1.2.1"
|
||||||
|
}
|
||||||
|
}
|
9
preact.config.js
Normal file
9
preact.config.js
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
const DefinePlugin = require("webpack/lib/DefinePlugin");
|
||||||
|
|
||||||
|
|
||||||
|
export default function (config, env, helpers) {
|
||||||
|
// noinspection JSUnresolvedVariable
|
||||||
|
config.resolve.alias["react"] = "preact/compat";
|
||||||
|
// noinspection JSUnresolvedVariable
|
||||||
|
config.resolve.alias["react-dom"] = "preact/compat";
|
||||||
|
};
|
25
src/components/Elements/Box.js
Normal file
25
src/components/Elements/Box.js
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
import style from "./Box.less";
|
||||||
|
|
||||||
|
export const BoxColors = Object.freeze({
|
||||||
|
RED: style.red,
|
||||||
|
ORANGE: style.orange,
|
||||||
|
YELLOW: style.yellow,
|
||||||
|
LIME: style.lime,
|
||||||
|
CYAN: style.cyan,
|
||||||
|
BLUE: style.blue,
|
||||||
|
MAGENTA: style.magenta,
|
||||||
|
DEFAULT: style.default
|
||||||
|
})
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
let color = BoxColors.DEFAULT;
|
||||||
|
if(props.color) {
|
||||||
|
color = props.color;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div class={style.box + " " + color}>
|
||||||
|
{props.children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
49
src/components/Elements/Box.less
Normal file
49
src/components/Elements/Box.less
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.box {
|
||||||
|
padding: 8px;
|
||||||
|
border-radius: 4px;
|
||||||
|
margin: 4px;
|
||||||
|
height: calc(100% - 8px);
|
||||||
|
min-width: 256px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.default {
|
||||||
|
background-color: fade(@fg, 5%);
|
||||||
|
color: @fg;
|
||||||
|
}
|
||||||
|
|
||||||
|
.red {
|
||||||
|
background-color: fade(@red, 5%);
|
||||||
|
color: @red;
|
||||||
|
}
|
||||||
|
|
||||||
|
.orange {
|
||||||
|
background-color: fade(@orange, 5%);
|
||||||
|
color: @orange;
|
||||||
|
}
|
||||||
|
|
||||||
|
.yellow {
|
||||||
|
background-color: fade(@yellow, 5%);
|
||||||
|
color: @yellow;
|
||||||
|
}
|
||||||
|
|
||||||
|
.lime {
|
||||||
|
background-color: fade(@lime, 5%);
|
||||||
|
color: @lime;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cyan {
|
||||||
|
background-color: fade(@cyan, 5%);
|
||||||
|
color: @cyan;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blue {
|
||||||
|
background-color: fade(@blue, 5%);
|
||||||
|
color: @blue;
|
||||||
|
}
|
||||||
|
|
||||||
|
.magenta {
|
||||||
|
background-color: fade(@magenta, 5%);
|
||||||
|
color: @magenta;
|
||||||
|
}
|
5
src/components/Elements/Image.js
Normal file
5
src/components/Elements/Image.js
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
export default function(props) {
|
||||||
|
return (
|
||||||
|
<a href={props.src} title={props.alt} target={"_blank"}><img src={props.src} alt={props.alt}/></a>
|
||||||
|
)
|
||||||
|
}
|
15
src/components/Elements/Panel.js
Normal file
15
src/components/Elements/Panel.js
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
import style from "./Panel.less";
|
||||||
|
import Box from "./Box";
|
||||||
|
|
||||||
|
export default function(props) {
|
||||||
|
return (
|
||||||
|
<Box color={props.color}>
|
||||||
|
<h3 class={style.title}>
|
||||||
|
{props.title}
|
||||||
|
</h3>
|
||||||
|
<div class={style.contents}>
|
||||||
|
{props.children}
|
||||||
|
</div>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
10
src/components/Elements/Panel.less
Normal file
10
src/components/Elements/Panel.less
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-family: @title;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.contents {
|
||||||
|
font-family: @text;
|
||||||
|
}
|
15
src/components/Elements/Section.js
Normal file
15
src/components/Elements/Section.js
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
import Split from "../Layout/Split";
|
||||||
|
import {Fragment} from "preact";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<h2>
|
||||||
|
{props.title}
|
||||||
|
</h2>
|
||||||
|
<Split>
|
||||||
|
{props.children}
|
||||||
|
</Split>
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
}
|
9
src/components/Elements/TablePanel.js
Normal file
9
src/components/Elements/TablePanel.js
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
import style from "./TablePanel.less";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
return (
|
||||||
|
<table class={style.tablepanel}>
|
||||||
|
{props.children}
|
||||||
|
</table>
|
||||||
|
);
|
||||||
|
}
|
6
src/components/Elements/TablePanel.less
Normal file
6
src/components/Elements/TablePanel.less
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.tablepanel {
|
||||||
|
margin: 4px;
|
||||||
|
width: calc(100% - 8px);
|
||||||
|
}
|
97
src/components/Elements/Timer.js
Normal file
97
src/components/Elements/Timer.js
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
import {Component} from 'preact'
|
||||||
|
import style from "./Timer.less"
|
||||||
|
|
||||||
|
|
||||||
|
export default class Timer extends Component {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
this.state = {
|
||||||
|
"now": Date.now()
|
||||||
|
};
|
||||||
|
this.timer = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
this.timer = setInterval(() => {
|
||||||
|
this.setState({"now": Date.now()})
|
||||||
|
}, 1000)
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
if(this.timer !== null) {
|
||||||
|
clearInterval(this.timer)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
let dateTo = "Unknown date";
|
||||||
|
let className = style.timer;
|
||||||
|
|
||||||
|
let parts = {
|
||||||
|
milliseconds: "?",
|
||||||
|
seconds: "?",
|
||||||
|
minutes: "?",
|
||||||
|
hours: "?",
|
||||||
|
days: "?",
|
||||||
|
};
|
||||||
|
|
||||||
|
if(this.props.to) {
|
||||||
|
dateTo = new Date(this.props.to);
|
||||||
|
let timeLeft = dateTo - this.state.now;
|
||||||
|
|
||||||
|
if(timeLeft > 0) {
|
||||||
|
parts = {
|
||||||
|
milliseconds: timeLeft % 1000,
|
||||||
|
seconds: Math.floor(timeLeft / 1000) % 60,
|
||||||
|
minutes: Math.floor(timeLeft / 60000) % 60,
|
||||||
|
hours: Math.floor(timeLeft / 3600000) % 24,
|
||||||
|
days: Math.floor(timeLeft / 86400000),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
parts = {
|
||||||
|
milliseconds: 0,
|
||||||
|
seconds: 0,
|
||||||
|
minutes: 0,
|
||||||
|
hours: 0,
|
||||||
|
days: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
className += " " + style.expired;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
className += " " + style.unknown;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div class={className} title={dateTo}>
|
||||||
|
<div class={style.days + " " + style.count}>
|
||||||
|
{parts.days}
|
||||||
|
</div>
|
||||||
|
<div className={style.days + " " + style.text}>
|
||||||
|
giorni
|
||||||
|
</div>
|
||||||
|
<div class={style.hours + " " + style.count}>
|
||||||
|
{parts.hours}
|
||||||
|
</div>
|
||||||
|
<div className={style.hours + " " + style.text}>
|
||||||
|
ore
|
||||||
|
</div>
|
||||||
|
<div class={style.minutes + " " + style.count}>
|
||||||
|
{parts.minutes}
|
||||||
|
</div>
|
||||||
|
<div className={style.minutes + " " + style.text}>
|
||||||
|
minuti
|
||||||
|
</div>
|
||||||
|
<div class={style.seconds + " " + style.count}>
|
||||||
|
{parts.seconds}
|
||||||
|
</div>
|
||||||
|
<div class={style.seconds + " " + style.text}>
|
||||||
|
secondi
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
57
src/components/Elements/Timer.less
Normal file
57
src/components/Elements/Timer.less
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
@import "../../styles/constants";
|
||||||
|
|
||||||
|
.timer {
|
||||||
|
display: grid;
|
||||||
|
text-align: center;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
margin-top: 8px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
padding: 8px;
|
||||||
|
grid-template-columns: 80px 80px 80px 80px;
|
||||||
|
border: 2px solid @plusplus;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.days {
|
||||||
|
grid-column: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hours {
|
||||||
|
grid-column: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.minutes {
|
||||||
|
grid-column: 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.seconds {
|
||||||
|
grid-column: 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.count {
|
||||||
|
grid-row: 1;
|
||||||
|
font-size: xx-large;
|
||||||
|
color: @accent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.text {
|
||||||
|
grid-row: 2;
|
||||||
|
font-size: small;
|
||||||
|
}
|
||||||
|
|
||||||
|
.unknown {
|
||||||
|
color: @magenta;
|
||||||
|
|
||||||
|
.count {
|
||||||
|
color: @magenta;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.expired {
|
||||||
|
color: @red;
|
||||||
|
|
||||||
|
.count {
|
||||||
|
color: @red;
|
||||||
|
}
|
||||||
|
}
|
10
src/components/Elements/Todo.js
Normal file
10
src/components/Elements/Todo.js
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
import style from "./Todo.less";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
if(process.env.NODE_ENV === "development") {
|
||||||
|
return <span class={style.todo}>{props.children}</span>;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
9
src/components/Elements/Todo.less
Normal file
9
src/components/Elements/Todo.less
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.todo {
|
||||||
|
border: 1px yellow solid;
|
||||||
|
border-radius: 2px;
|
||||||
|
padding: 1px;
|
||||||
|
background-color: black;
|
||||||
|
color: yellow;
|
||||||
|
}
|
18
src/components/Layout/Footer.js
Normal file
18
src/components/Layout/Footer.js
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
import style from './Footer.less';
|
||||||
|
import { Component } from 'preact';
|
||||||
|
|
||||||
|
export default function(props) {
|
||||||
|
return (
|
||||||
|
<div class={style.footer}>
|
||||||
|
© {new Date().getFullYear()}
|
||||||
|
-
|
||||||
|
<a href={"https://steffo.eu/"}>Stefano Pigozzi</a>
|
||||||
|
-
|
||||||
|
<a href="https://creativecommons.org/licenses/by-sa/4.0/">CC BY-SA 4.0</a>
|
||||||
|
-
|
||||||
|
<a href="https://github.com/Steffo99/appuntiweb">uni.steffo.eu {process.env.RELEASE}</a>
|
||||||
|
-
|
||||||
|
<a href={"https://ko-fi.com/steffo"}>Supportami</a>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
8
src/components/Layout/Footer.less
Normal file
8
src/components/Layout/Footer.less
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.footer {
|
||||||
|
margin-top: 8px;
|
||||||
|
color: @accent;
|
||||||
|
text-align: center;
|
||||||
|
font-size: x-small;
|
||||||
|
}
|
28
src/components/Layout/Split.js
Normal file
28
src/components/Layout/Split.js
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
import style from "./Split.less";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
let children;
|
||||||
|
|
||||||
|
if(Array.isArray(props.children)) {
|
||||||
|
children = props.children.map(element => {
|
||||||
|
return (
|
||||||
|
<div class={style.splitchild}>
|
||||||
|
{element}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
children = (
|
||||||
|
<div class={style.splitchild}>
|
||||||
|
{props.children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div class={style.split}>
|
||||||
|
<div class={style.splitparent}>{children}</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
16
src/components/Layout/Split.less
Normal file
16
src/components/Layout/Split.less
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.split {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.splitparent {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.splitchild {
|
||||||
|
flex-grow: 1;
|
||||||
|
flex-shrink: 0;
|
||||||
|
flex-basis: 0;
|
||||||
|
}
|
7
src/components/Rendering/BLatex.js
Normal file
7
src/components/Rendering/BLatex.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import Latex from "./Latex";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
return (
|
||||||
|
<Latex inline={false}>{props.children}</Latex>
|
||||||
|
);
|
||||||
|
}
|
11
src/components/Rendering/Code.js
Normal file
11
src/components/Rendering/Code.js
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
import SyntaxHighlighter from 'react-syntax-highlighter'
|
||||||
|
import {monokai} from "react-syntax-highlighter/dist/cjs/styles/hljs";
|
||||||
|
import stripTabs from "../../utils/stripTabs";
|
||||||
|
|
||||||
|
export default function(props) {
|
||||||
|
return (
|
||||||
|
<SyntaxHighlighter language={props.language ? props.language : "plaintext"} style={monokai}>
|
||||||
|
{stripTabs(String(props.children))}
|
||||||
|
</SyntaxHighlighter>
|
||||||
|
)
|
||||||
|
}
|
7
src/components/Rendering/ILatex.js
Normal file
7
src/components/Rendering/ILatex.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import Latex from "./Latex";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
return (
|
||||||
|
<Latex inline={true}>{props.children}</Latex>
|
||||||
|
);
|
||||||
|
}
|
39
src/components/Rendering/Latex.js
Normal file
39
src/components/Rendering/Latex.js
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
import style from './Latex.less';
|
||||||
|
import {useContext} from "preact/hooks";
|
||||||
|
import LatexRenderColor from "../../contexts/LatexRenderColor";
|
||||||
|
import LatexDefaultInline from "../../contexts/LatexDefaultInline";
|
||||||
|
|
||||||
|
export default function(props) {
|
||||||
|
// black, blue, brown, cyan, darkgray, gray, green, lightgray, lime, magenta, olive, orange, pink, purple, red, teal, violet, white, yellow
|
||||||
|
let renderColor = useContext(LatexRenderColor);
|
||||||
|
let defaultInline = useContext(LatexDefaultInline);
|
||||||
|
|
||||||
|
let is_inline;
|
||||||
|
if(props.inline === undefined) {
|
||||||
|
is_inline = defaultInline;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
is_inline = props.inline;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(is_inline) {
|
||||||
|
let equation = `\\inline {\\color{${renderColor}} ${props.children} }`;
|
||||||
|
return (
|
||||||
|
<img src={`https://latex.codecogs.com/svg.latex?${equation}`}
|
||||||
|
alt={props.children}
|
||||||
|
title={props.children}
|
||||||
|
class={style.latex + " " + style.inline}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
let equation = `{\\color{${renderColor}} ${props.children} }`;
|
||||||
|
return (
|
||||||
|
<img src={`https://latex.codecogs.com/svg.latex?${equation}`}
|
||||||
|
alt={props.children}
|
||||||
|
title={props.children}
|
||||||
|
class={style.latex + " " + style.block}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
14
src/components/Rendering/Latex.less
Normal file
14
src/components/Rendering/Latex.less
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.latex {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.inline {
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
|
||||||
|
.block {
|
||||||
|
display: block;
|
||||||
|
}
|
14
src/components/Rendering/Markdown.js
Normal file
14
src/components/Rendering/Markdown.js
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
import showdown from "showdown";
|
||||||
|
import stripTabs from "../../utils/stripTabs";
|
||||||
|
import style from "./Markdown.less";
|
||||||
|
|
||||||
|
export default function(props) {
|
||||||
|
let converter = new showdown.Converter({
|
||||||
|
"tables": true,
|
||||||
|
});
|
||||||
|
converter.setFlavor("github");
|
||||||
|
|
||||||
|
let html = converter.makeHtml(stripTabs(String(props.children)));
|
||||||
|
|
||||||
|
return <div class={style.markdown} dangerouslySetInnerHTML={{__html: html}}/>;
|
||||||
|
}
|
12
src/components/Rendering/Markdown.less
Normal file
12
src/components/Rendering/Markdown.less
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
@import "../../styles/constants.less";
|
||||||
|
|
||||||
|
.markdown {
|
||||||
|
h1, h2, h3, h4, h5, h6 {
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
padding-bottom: 2px;
|
||||||
|
border-bottom: 1px solid @plusplusplus;
|
||||||
|
}
|
||||||
|
}
|
9
src/components/Rendering/PLatex.js
Normal file
9
src/components/Rendering/PLatex.js
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
import BLatex from "./BLatex";
|
||||||
|
|
||||||
|
export default function (props) {
|
||||||
|
return (
|
||||||
|
<p>
|
||||||
|
<BLatex>{props.children}</BLatex>
|
||||||
|
</p>
|
||||||
|
);
|
||||||
|
}
|
3
src/contexts/LatexDefaultInline.js
Normal file
3
src/contexts/LatexDefaultInline.js
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
import {createContext} from "preact";
|
||||||
|
|
||||||
|
export default createContext(true);
|
3
src/contexts/LatexRenderColor.js
Normal file
3
src/contexts/LatexRenderColor.js
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
import {createContext} from "preact";
|
||||||
|
|
||||||
|
export default createContext("White");
|
64
src/index.js
Normal file
64
src/index.js
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
import Box from "./components/Elements/Box"
|
||||||
|
import {BoxColors} from "./components/Elements/Box";
|
||||||
|
import Image from "./components/Elements/Image"
|
||||||
|
import Panel from "./components/Elements/Panel";
|
||||||
|
import Section from "./components/Elements/Section";
|
||||||
|
import TablePanel from "./components/Elements/TablePanel";
|
||||||
|
import Timer from "./components/Elements/Timer";
|
||||||
|
import Todo from "./components/Elements/Todo";
|
||||||
|
import Footer from "./components/Layout/Footer";
|
||||||
|
import Split from "./components/Layout/Split";
|
||||||
|
import Code from "./components/Rendering/Code";
|
||||||
|
import Latex from "./components/Rendering/Latex";
|
||||||
|
import BLatex from "./components/Rendering/BLatex";
|
||||||
|
import ILatex from "./components/Rendering/ILatex";
|
||||||
|
import PLatex from "./components/Rendering/PLatex";
|
||||||
|
import LatexDefaultInline from "./contexts/LatexDefaultInline";
|
||||||
|
import LatexRenderColor from "./contexts/LatexRenderColor";
|
||||||
|
import Markdown from "./components/Rendering/Markdown";
|
||||||
|
|
||||||
|
export {
|
||||||
|
Box,
|
||||||
|
BoxColors,
|
||||||
|
Image,
|
||||||
|
Panel,
|
||||||
|
Section,
|
||||||
|
TablePanel,
|
||||||
|
Timer,
|
||||||
|
Todo,
|
||||||
|
Footer,
|
||||||
|
Split,
|
||||||
|
Code,
|
||||||
|
Latex,
|
||||||
|
BLatex,
|
||||||
|
ILatex,
|
||||||
|
PLatex,
|
||||||
|
LatexDefaultInline,
|
||||||
|
LatexRenderColor,
|
||||||
|
Markdown,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
import {Fragment} from "preact";
|
||||||
|
// noinspection JSUnusedGlobalSymbols
|
||||||
|
export default function() {
|
||||||
|
require("./styles/theme.less");
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<h1>
|
||||||
|
bluelib
|
||||||
|
</h1>
|
||||||
|
<Split>
|
||||||
|
<Box>
|
||||||
|
Hi. I'm a box!
|
||||||
|
</Box>
|
||||||
|
<Box color={BoxColors.RED}>
|
||||||
|
Hi. I'm a box, but I'm red!
|
||||||
|
</Box>
|
||||||
|
<Box color={BoxColors.LIME}>
|
||||||
|
Hi. I was here third!
|
||||||
|
</Box>
|
||||||
|
</Split>
|
||||||
|
</Fragment>
|
||||||
|
)
|
||||||
|
}
|
33
src/styles/constants.less
Normal file
33
src/styles/constants.less
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
@bg: #0d193b;
|
||||||
|
@bg-light: #142245;
|
||||||
|
@bg-lighter: #1c2b4f;
|
||||||
|
@bg-lightest: #233358;
|
||||||
|
@fg: #a0ccff;
|
||||||
|
@accent: #ffffff;
|
||||||
|
|
||||||
|
@link: #00caca;
|
||||||
|
@linkhover: #00ffff;
|
||||||
|
@linkactive: #a0ffff;
|
||||||
|
|
||||||
|
@plus: fade(@fg, 5%);
|
||||||
|
@plusplus: fade(@fg, 10%);
|
||||||
|
@plusplusplus: fade(@fg, 15%);
|
||||||
|
@plusplusplusplus: fade(@fg, 20%);
|
||||||
|
|
||||||
|
@red: #ff7d7d;
|
||||||
|
@orange: #ffbb7d;
|
||||||
|
@yellow: #ffff7d;
|
||||||
|
@lime: #7dff7d;
|
||||||
|
@cyan: #7dffff;
|
||||||
|
@blue: #7d7dff;
|
||||||
|
@magenta: #ff7dff;
|
||||||
|
|
||||||
|
@disabledfg: #808080;
|
||||||
|
@disabledbg: #1f1f1f;
|
||||||
|
|
||||||
|
@sans: "Arial", sans-serif;
|
||||||
|
@text: "Calibri", sans-serif;
|
||||||
|
@title: "Verdana", sans-serif;
|
||||||
|
@mono: "Consolas", "SFMono-Regular", "Liberation Mono", "Menlo", monospace;
|
||||||
|
|
||||||
|
@example: #d3a1ff;
|
154
src/styles/theme.less
Normal file
154
src/styles/theme.less
Normal file
|
@ -0,0 +1,154 @@
|
||||||
|
@import "constants.less";
|
||||||
|
|
||||||
|
* {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: @bg;
|
||||||
|
color: @fg;
|
||||||
|
font-family: @sans;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1, h2, h3, h4, h5, h6 {
|
||||||
|
margin-top: 4px;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
color: @accent;
|
||||||
|
font-family: @title;
|
||||||
|
font-weight: normal;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
// By default h1 are as large as h2
|
||||||
|
h1 {
|
||||||
|
font-size: xx-large;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: @link;
|
||||||
|
text-decoration: none;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: @linkhover;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:active {
|
||||||
|
color: @linkactive;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
img, iframe {
|
||||||
|
max-width: 100%;
|
||||||
|
max-height: 300px;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre, code {
|
||||||
|
font-family: @mono;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
color: @fg;
|
||||||
|
border-left: 3px solid @plusplusplusplus;
|
||||||
|
background-color: @plus;
|
||||||
|
padding: 4px 4px 4px 8px;
|
||||||
|
margin: 8px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type="text"], input[type="password"] {
|
||||||
|
color: @fg;
|
||||||
|
background-color: @bg;
|
||||||
|
border: 1px solid @plusplus;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 4px;
|
||||||
|
font-size: medium;
|
||||||
|
|
||||||
|
&:disabled, &.disabled {
|
||||||
|
color: @disabledfg;
|
||||||
|
background-color: @disabledbg;
|
||||||
|
border-style: dotted;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
color: @fg;
|
||||||
|
background-color: @bg;
|
||||||
|
border: 1px solid @plusplus;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 4px;
|
||||||
|
font-size: medium;
|
||||||
|
|
||||||
|
&:hover, &.hover {
|
||||||
|
background-color: @plusplus;
|
||||||
|
border: 1px solid @fg;
|
||||||
|
color: @fg;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:active, &.active {
|
||||||
|
background-color: fade(@accent, 20%);
|
||||||
|
border: 1px solid @accent;
|
||||||
|
color: @accent;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:disabled, &.disabled {
|
||||||
|
color: @disabledfg;
|
||||||
|
background-color: @disabledbg;
|
||||||
|
border-style: dotted;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
border: 1px solid @plusplusplusplus;
|
||||||
|
margin-top: 24px;
|
||||||
|
margin-bottom: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-spacing: 0;
|
||||||
|
border: 2px solid @plusplus;
|
||||||
|
background-color: @plus;
|
||||||
|
border-collapse: collapse;
|
||||||
|
|
||||||
|
thead, tbody {
|
||||||
|
th, td {
|
||||||
|
padding: 4px;
|
||||||
|
border: 1px solid @plusplus;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
thead {
|
||||||
|
background-color: @plusplus;
|
||||||
|
color: @accent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
li {
|
||||||
|
margin: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
p:first-child {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
p:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
b {
|
||||||
|
color: @accent;
|
||||||
|
}
|
||||||
|
|
||||||
|
abbr {
|
||||||
|
cursor: help;
|
||||||
|
}
|
||||||
|
|
||||||
|
aside {
|
||||||
|
margin: 4px 0;
|
||||||
|
padding: 4px;
|
||||||
|
font-size: smaller;
|
||||||
|
background-color: @plus;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
25
src/utils/stripTabs.js
Normal file
25
src/utils/stripTabs.js
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
export default function(input) {
|
||||||
|
let indent_regex = /^[ \t]+/;
|
||||||
|
|
||||||
|
let lines = input.split("\n").filter((line) => {
|
||||||
|
return line !== "";
|
||||||
|
});
|
||||||
|
let match = null;
|
||||||
|
|
||||||
|
for(let i = 0; i < lines.length; i++) {
|
||||||
|
match = indent_regex.exec(lines[i]);
|
||||||
|
if(match !== null) break;
|
||||||
|
}
|
||||||
|
|
||||||
|
let start;
|
||||||
|
if(match === null) {
|
||||||
|
start = 0;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
start = match[0].length;
|
||||||
|
}
|
||||||
|
|
||||||
|
return lines.map((line) => {
|
||||||
|
return line.substr(start);
|
||||||
|
}).join("\n");
|
||||||
|
}
|
Loading…
Reference in a new issue