Updated to work with live site
Added components from somewhere fixed some css background image
This commit is contained in:
parent
dc6580e21c
commit
dce18f97d1
@ -13,8 +13,7 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"compression": "^1.7.4",
|
"compression": "^1.7.4",
|
||||||
"polka": "next",
|
"polka": "next"
|
||||||
"sirv": "^1.0.6"
|
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"npm-run-all": "^4.1.5",
|
"npm-run-all": "^4.1.5",
|
||||||
@ -31,6 +30,7 @@
|
|||||||
"@rollup/plugin-replace": "^2.3.3",
|
"@rollup/plugin-replace": "^2.3.3",
|
||||||
"rollup": "^2.26.4",
|
"rollup": "^2.26.4",
|
||||||
"rollup-plugin-svelte": "^5.2.3",
|
"rollup-plugin-svelte": "^5.2.3",
|
||||||
"rollup-plugin-terser": "^7.0.0"
|
"rollup-plugin-terser": "^7.0.0",
|
||||||
|
"sirv": "^1.0.6"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
94
src/components/Barcode.svelte
Normal file
94
src/components/Barcode.svelte
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
<script>
|
||||||
|
import {onMount, tick, beforeUpdate} from 'svelte' ;
|
||||||
|
|
||||||
|
let barcodeReady = false;
|
||||||
|
let mounted = false;
|
||||||
|
|
||||||
|
onMount(() => {
|
||||||
|
// The payment-form is ready.
|
||||||
|
mounted = true;
|
||||||
|
//if (barcodeReady) {
|
||||||
|
loadBarcodeElements();
|
||||||
|
//}
|
||||||
|
});
|
||||||
|
|
||||||
|
//export const
|
||||||
|
|
||||||
|
const target = (bc) => bc.length < 9 ? 7 : 12;
|
||||||
|
|
||||||
|
let error;
|
||||||
|
|
||||||
|
$: format = barcode.length < 9 ? "EAN8" : "EAN13";
|
||||||
|
|
||||||
|
|
||||||
|
function padBarcode(bc, length, pad = 0) {
|
||||||
|
return bc;
|
||||||
|
while (bc.length < length) {
|
||||||
|
bc = pad + bc.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
return bc;
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadBarcodeElements = () => {
|
||||||
|
clearCanvas(document.getElementById("barcode").getContext("2d"));
|
||||||
|
error = null;
|
||||||
|
if (barcode != null && barcode.length > 0) {
|
||||||
|
//bwipjs.toCanvas("barcode", {bcid:format, text: padBarcode(barcode, target(barcode))});
|
||||||
|
try {
|
||||||
|
|
||||||
|
const x = JsBarcode("#barcode", padBarcode(barcode, target(barcode)), {format: format});
|
||||||
|
console.log(x);
|
||||||
|
} catch {
|
||||||
|
error = "Invalid barcode"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export let barcode = "";
|
||||||
|
|
||||||
|
beforeUpdate(async () => {
|
||||||
|
if (mounted) {
|
||||||
|
loadBarcodeElements();
|
||||||
|
}
|
||||||
|
await tick();
|
||||||
|
});
|
||||||
|
|
||||||
|
export const appendCheckDigit = () => {
|
||||||
|
if (barcode.length === 7 || barcode.length === 12) {
|
||||||
|
const check = eanCheckDigit(barcode);
|
||||||
|
//if (barcode[barcode.length-1] !== check)
|
||||||
|
//{
|
||||||
|
barcode += check;
|
||||||
|
//}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
function clearCanvas(context, canvas = context.canvas) {
|
||||||
|
context.clearRect(0, 0, canvas.width, canvas.height);
|
||||||
|
const w = canvas.width;
|
||||||
|
canvas.width = 1;
|
||||||
|
canvas.width = w;
|
||||||
|
}
|
||||||
|
|
||||||
|
// definitely a hack.
|
||||||
|
export const eanCheckDigit = (s) => {
|
||||||
|
let x = JsBarcode("#barcode", s, {format: format});
|
||||||
|
|
||||||
|
//EAN barcodes by default have 5 parts (prefix -> left part -> middle seperator -> right part -> suffix)
|
||||||
|
// we only want left & right part (which contain a text
|
||||||
|
let calculated = "";
|
||||||
|
x._encodings[0].filter(e => e.text !== "").forEach(e => calculated += e.text);
|
||||||
|
console.log(calculated);
|
||||||
|
return calculated[calculated.length - 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
|
<canvas id="barcode" width="50%" height="100%"></canvas>
|
||||||
|
{#if error}
|
||||||
|
<p>Error: {error}</p>
|
||||||
|
{/if}
|
||||||
|
|
36
src/components/Link.svelte
Normal file
36
src/components/Link.svelte
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
<script>
|
||||||
|
import {getContext} from 'svelte';
|
||||||
|
import {writable} from 'svelte/store';
|
||||||
|
|
||||||
|
const navigate = getContext('navigate');
|
||||||
|
const navaid = getContext('navaid');
|
||||||
|
|
||||||
|
const base = navaid && navaid.base || writable('/');
|
||||||
|
const library = navaid && navaid.library || writable(null);
|
||||||
|
let props;
|
||||||
|
let href;
|
||||||
|
|
||||||
|
export let append;
|
||||||
|
|
||||||
|
$: {
|
||||||
|
props = {...$$props};
|
||||||
|
href = props.href;
|
||||||
|
delete props.href;
|
||||||
|
href = ($library && $library.prefix ? $library.prefix : '') + (!append ? $base : '') + href.replace(/^\//, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
const click = () => {
|
||||||
|
navigate(href);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
span {
|
||||||
|
text-decoration: none;
|
||||||
|
padding: 1rem 0.5rem;
|
||||||
|
display: block;
|
||||||
|
color: #005de6;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<span {...props} on:click={click}><slot></slot></span>
|
249
src/components/Modal.svelte
Normal file
249
src/components/Modal.svelte
Normal file
@ -0,0 +1,249 @@
|
|||||||
|
<script>
|
||||||
|
import {setContext as baseSetContext, createEventDispatcher} from 'svelte';
|
||||||
|
import {fade} from 'svelte/transition';
|
||||||
|
|
||||||
|
export let key = 'simple-modal';
|
||||||
|
export let closeButton = true;
|
||||||
|
export let closeOnEsc = true;
|
||||||
|
export let closeOnOuterClick = true;
|
||||||
|
export let transitionBg = fade;
|
||||||
|
export let transitionBgProps = {duration: 250};
|
||||||
|
export let transitionWindow = transitionBg;
|
||||||
|
export let transitionWindowProps = transitionBgProps;
|
||||||
|
export let styleBg = {top: 0, left: 0};
|
||||||
|
export let styleWindow = {};
|
||||||
|
export let styleContent = {};
|
||||||
|
export let setContext = baseSetContext;
|
||||||
|
|
||||||
|
let Component = null;
|
||||||
|
let props = null;
|
||||||
|
|
||||||
|
let background;
|
||||||
|
let wrap;
|
||||||
|
let customStyleBg = {};
|
||||||
|
let customStyleWindow = {};
|
||||||
|
let customStyleContent = {};
|
||||||
|
|
||||||
|
let Callback = (props) => {
|
||||||
|
};
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
const camelCaseToDash = str => str
|
||||||
|
.replace(/([a-zA-Z])(?=[A-Z])/g, '$1-').toLowerCase();
|
||||||
|
|
||||||
|
const toCssString = (props) => Object.keys(props)
|
||||||
|
.reduce((str, key) => `${str}; ${camelCaseToDash(key)}: ${props[key]}`, '');
|
||||||
|
|
||||||
|
$: cssBg = toCssString(Object.assign({}, styleBg, customStyleBg));
|
||||||
|
$: cssWindow = toCssString(Object.assign({}, styleWindow, customStyleWindow));
|
||||||
|
$: cssContent = toCssString(Object.assign({}, styleContent, customStyleContent));
|
||||||
|
|
||||||
|
const open = (NewComponent, newProps = {}, style = {bg: {}, window: {}, content: {}}, newCallback = () => {
|
||||||
|
}) => {
|
||||||
|
Component = NewComponent;
|
||||||
|
props = newProps;
|
||||||
|
Callback = newCallback;
|
||||||
|
customStyleBg = style.bg || {};
|
||||||
|
customStyleWindow = style.window || {};
|
||||||
|
customStyleContent = style.content || {};
|
||||||
|
};
|
||||||
|
|
||||||
|
const close = () => {
|
||||||
|
Callback(props);
|
||||||
|
Component = null;
|
||||||
|
props = null;
|
||||||
|
customStyleBg = {};
|
||||||
|
customStyleWindow = {};
|
||||||
|
customStyleContent = {};
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleKeyup = ({key}) => {
|
||||||
|
if (closeOnEsc && Component && key === 'Escape') {
|
||||||
|
event.preventDefault();
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleOuterClick = (event) => {
|
||||||
|
if (
|
||||||
|
closeOnOuterClick && (
|
||||||
|
event.target === background || event.target === wrap
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
event.preventDefault();
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
setContext(key, {open, close});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
|
||||||
|
** LICENSE **
|
||||||
|
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2019 Fritz Lekschas
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
|
|
||||||
|
https://github.com/flekschas/svelte-simple-moda
|
||||||
|
-->
|
||||||
|
|
||||||
|
<style>
|
||||||
|
* {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 1000;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
background: rgba(0, 0, 0, 0.66);
|
||||||
|
}
|
||||||
|
|
||||||
|
.window-wrap {
|
||||||
|
position: relative;
|
||||||
|
margin: 2rem;
|
||||||
|
max-height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.window {
|
||||||
|
position: relative;
|
||||||
|
width: 40rem;
|
||||||
|
max-width: 100%;
|
||||||
|
max-height: 100%;
|
||||||
|
margin: 2rem auto;
|
||||||
|
color: black;
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
position: relative;
|
||||||
|
padding: 1rem;
|
||||||
|
max-height: calc(100vh - 4rem);
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close {
|
||||||
|
display: block;
|
||||||
|
box-sizing: border-box;
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1000;
|
||||||
|
top: 1rem;
|
||||||
|
right: 1rem;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
width: 1.5rem;
|
||||||
|
height: 1.5rem;
|
||||||
|
border: 0;
|
||||||
|
color: black;
|
||||||
|
border-radius: 1.5rem;
|
||||||
|
background: white;
|
||||||
|
box-shadow: 0 0 0 1px black;
|
||||||
|
transition: transform 0.2s cubic-bezier(0.25, 0.1, 0.25, 1),
|
||||||
|
background 0.2s cubic-bezier(0.25, 0.1, 0.25, 1);
|
||||||
|
-webkit-appearance: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:before, .close:after {
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
box-sizing: border-box;
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
width: 1rem;
|
||||||
|
height: 1px;
|
||||||
|
background: black;
|
||||||
|
transform-origin: center;
|
||||||
|
transition: height 0.2s cubic-bezier(0.25, 0.1, 0.25, 1),
|
||||||
|
background 0.2s cubic-bezier(0.25, 0.1, 0.25, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:before {
|
||||||
|
-webkit-transform: translate(0, -50%) rotate(45deg);
|
||||||
|
-moz-transform: translate(0, -50%) rotate(45deg);
|
||||||
|
transform: translate(0, -50%) rotate(45deg);
|
||||||
|
left: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:after {
|
||||||
|
-webkit-transform: translate(0, -50%) rotate(-45deg);
|
||||||
|
-moz-transform: translate(0, -50%) rotate(-45deg);
|
||||||
|
transform: translate(0, -50%) rotate(-45deg);
|
||||||
|
left: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:hover {
|
||||||
|
background: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:hover:before, .close:hover:after {
|
||||||
|
height: 2px;
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:focus {
|
||||||
|
border-color: #3399ff;
|
||||||
|
box-shadow: 0 0 0 2px #3399ff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:active {
|
||||||
|
transform: scale(0.9);
|
||||||
|
}
|
||||||
|
|
||||||
|
.close:hover, .close:focus, .close:active {
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<svelte:window on:keyup={handleKeyup}/>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
{#if Component}
|
||||||
|
<div
|
||||||
|
class="bg"
|
||||||
|
on:click={handleOuterClick}
|
||||||
|
bind:this={background}
|
||||||
|
transition:transitionBg={transitionBgProps}
|
||||||
|
style={cssBg}>
|
||||||
|
<div class="window-wrap" bind:this={wrap}>
|
||||||
|
<div
|
||||||
|
class="window"
|
||||||
|
transition:transitionWindow={transitionWindowProps}
|
||||||
|
style={cssWindow}>
|
||||||
|
{#if closeButton}
|
||||||
|
<button on:click={close} class="close"></button>
|
||||||
|
{/if}
|
||||||
|
<div class="content" style={cssContent}>
|
||||||
|
<Component {...props} {close}/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
<slot></slot>
|
||||||
|
</div>
|
56
src/components/Sidebar.svelte
Normal file
56
src/components/Sidebar.svelte
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
{#if show}
|
||||||
|
<div class="overlay" data-close on:click={overlay_click}>
|
||||||
|
<nav transition:fly={{x: -250, opacity: 1}}>
|
||||||
|
<slot>
|
||||||
|
<button on:click={click}>About</button>
|
||||||
|
</slot>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<Modal bind:show={modal_show}/>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {fly} from 'svelte/transition';
|
||||||
|
import Modal from './Modal.svelte';
|
||||||
|
|
||||||
|
const click = () => {
|
||||||
|
modal_show = true;
|
||||||
|
show = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
function overlay_click(e) {
|
||||||
|
if ('close' in e.target.dataset)
|
||||||
|
show = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
export let show = false;
|
||||||
|
let modal_show = false;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background: rgba(0, 0, 0, 0);
|
||||||
|
z-index: 10;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
|
padding: 2rem 1rem 0.6rem;
|
||||||
|
border-right: 1px solid #aaa;
|
||||||
|
background: #fff;
|
||||||
|
overflow-y: auto;
|
||||||
|
width: 20rem;
|
||||||
|
}
|
||||||
|
</style>
|
249
src/components/SortableList.svelte
Normal file
249
src/components/SortableList.svelte
Normal file
@ -0,0 +1,249 @@
|
|||||||
|
<script>
|
||||||
|
|
||||||
|
import {quintOut} from "svelte/easing";
|
||||||
|
import {crossfade} from "svelte/transition";
|
||||||
|
import {flip} from "svelte/animate";
|
||||||
|
// FLIP ANIMATION
|
||||||
|
const [send, receive] = crossfade({
|
||||||
|
duration: d => Math.sqrt(d * 200),
|
||||||
|
fallback(node, params) {
|
||||||
|
const style = getComputedStyle(node);
|
||||||
|
const transform = style.transform === "none" ? "" : style.transform;
|
||||||
|
return {
|
||||||
|
duration: 600,
|
||||||
|
easing: quintOut,
|
||||||
|
css: t => `
|
||||||
|
transform: ${transform} scale(${t});
|
||||||
|
opacity: ${t}
|
||||||
|
`
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// DRAG AND DROP
|
||||||
|
let isOver = false;
|
||||||
|
let isDragging = false;
|
||||||
|
const getDraggedParent = (node) => {
|
||||||
|
return node.dataset && node.dataset.index
|
||||||
|
? node.dataset
|
||||||
|
: getDraggedParent(node.parentNode)
|
||||||
|
};
|
||||||
|
const click = ev => {
|
||||||
|
dispatch("slotclick", {
|
||||||
|
source: ev.target.parentElement.dataset.index,
|
||||||
|
sourceItem: list[ev.target.parentElement.dataset.index]
|
||||||
|
});
|
||||||
|
};
|
||||||
|
const start = ev => {
|
||||||
|
isDragging = true;
|
||||||
|
ev.dataTransfer.setData("source", ev.target.dataset.index);
|
||||||
|
};
|
||||||
|
const over = ev => {
|
||||||
|
ev.preventDefault();
|
||||||
|
let dragged = getDraggedParent(ev.target);
|
||||||
|
if (isOver !== dragged.id) isOver = JSON.parse(dragged.id);
|
||||||
|
};
|
||||||
|
const leave = ev => {
|
||||||
|
let dragged = getDraggedParent(ev.target);
|
||||||
|
if (isOver === dragged.id) isOver = false;
|
||||||
|
};
|
||||||
|
const drop = ev => {
|
||||||
|
isOver = false;
|
||||||
|
ev.preventDefault();
|
||||||
|
let dragged = getDraggedParent(ev.target);
|
||||||
|
let from = ev.dataTransfer.getData("source");
|
||||||
|
let to = dragged.index;
|
||||||
|
reorder({from, to});
|
||||||
|
isDragging = false;
|
||||||
|
};
|
||||||
|
// DISPATCH REORDER
|
||||||
|
import {createEventDispatcher, onMount, afterUpdate, beforeUpdate, tick} from "svelte";
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher();
|
||||||
|
const reorder = ({from, to}) => {
|
||||||
|
let newList = [...list];
|
||||||
|
let temp = newList[from];
|
||||||
|
newList[from] = newList[to];
|
||||||
|
newList[to] = temp;
|
||||||
|
//newList[from] = [newList[to], (newList[to] = newList[from])][0];
|
||||||
|
dispatch("sort", {data: newList, from: from, to: to});
|
||||||
|
};
|
||||||
|
// UTILS
|
||||||
|
const getKey = item => (key ? item[key] : item);
|
||||||
|
// PROPS
|
||||||
|
export let list;
|
||||||
|
export let key;
|
||||||
|
export let ulclass;
|
||||||
|
|
||||||
|
export let breakon;
|
||||||
|
|
||||||
|
export let breakComponent;
|
||||||
|
// return `<><li class='${breakClass}' style='height:100%; grid-column: span 5;'><div style='width:100%;height:inherit;'></li>`;
|
||||||
|
|
||||||
|
// export let breakPosition = "beforebegin";
|
||||||
|
let breakClass = "group-break";
|
||||||
|
|
||||||
|
const Elements = () => document.getElementsByClassName(key);
|
||||||
|
|
||||||
|
const removeOldBreaks = (parent) => {
|
||||||
|
if (breakon == null) return;
|
||||||
|
if (isDragging) return;
|
||||||
|
|
||||||
|
if (parent.length) {
|
||||||
|
const old = parent[0].getElementsByClassName(breakClass);
|
||||||
|
|
||||||
|
for (let i = 0; i < old.length; i = 0) {
|
||||||
|
old[i].remove(); // removing the node from the DOM also removes it from this array (WTF??)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let lastCollection;
|
||||||
|
|
||||||
|
let updatingBreakPoints = false;
|
||||||
|
|
||||||
|
const updateBreak = () => {
|
||||||
|
if (isDragging || updatingBreakPoints) return;
|
||||||
|
if (breakon == null) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
updatingBreakPoints = true;
|
||||||
|
|
||||||
|
let breaks = 0;
|
||||||
|
let last = null;
|
||||||
|
let carry = 0;
|
||||||
|
let elements = Elements();
|
||||||
|
|
||||||
|
|
||||||
|
for (let x = 0; x < list.length; x++) {
|
||||||
|
|
||||||
|
if (last != null) {
|
||||||
|
if (list[x][breakon] > last) {
|
||||||
|
|
||||||
|
if (elements.length) {
|
||||||
|
let targetIndex = x + breaks;
|
||||||
|
|
||||||
|
if (list.length !== elements[0].children.length - breaks) {
|
||||||
|
if (lastCollection)
|
||||||
|
if (lastCollection.length)
|
||||||
|
targetIndex += lastCollection.length;
|
||||||
|
}
|
||||||
|
// elements[0].children[targetIndex].insertAdjacentHTML(breakPosition, breakHtml(carry + 1));
|
||||||
|
breaks++;
|
||||||
|
carry = 0;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
carry++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
last = list[x][breakon] > last ? list[x][breakon] : last;
|
||||||
|
}
|
||||||
|
|
||||||
|
lastCollection = list;
|
||||||
|
} finally {
|
||||||
|
updatingBreakPoints = false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
//beforeUpdate(() => {updateBreak()});
|
||||||
|
|
||||||
|
let updating = false;
|
||||||
|
|
||||||
|
afterUpdate(async () => {
|
||||||
|
if (isDragging) return;
|
||||||
|
if (breakon == null) return;
|
||||||
|
|
||||||
|
updating = true;
|
||||||
|
removeOldBreaks(Elements());
|
||||||
|
await tick();
|
||||||
|
updateBreak();
|
||||||
|
|
||||||
|
updating = false;
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
const getClass = () => {
|
||||||
|
let ret = "";
|
||||||
|
if (key)
|
||||||
|
ret += ` ${key}`;
|
||||||
|
if (ulclass)
|
||||||
|
ret += ` ${ulclass}`;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
|
||||||
|
ul {
|
||||||
|
display: inline-grid;
|
||||||
|
list-style: none;
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
li {
|
||||||
|
display: inline-block;
|
||||||
|
border: 2px dotted transparent;
|
||||||
|
transition: border 0.1s linear;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
margin: .5%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.startsNewGroup {
|
||||||
|
grid-column-start: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.small {
|
||||||
|
grid-column-end: span 1;
|
||||||
|
grid-row: span 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.medium {
|
||||||
|
grid-column-end: span 2;
|
||||||
|
grid-row: span 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.large {
|
||||||
|
grid-column-end: span 2;
|
||||||
|
/*grid-row: span 2; large buttons are double height, but they also push the row below them down (they dont fit around eachother like css-grid) */
|
||||||
|
}
|
||||||
|
|
||||||
|
.over {
|
||||||
|
border-color: rgba(48, 12, 200, 0.2);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
{#if list && list.length}
|
||||||
|
<ul class={getClass()}>
|
||||||
|
{#each list as item, index (getKey(item))}
|
||||||
|
<li
|
||||||
|
class:small={item.displayTileSize === 2}
|
||||||
|
class:medium={item.displayTileSize === 3}
|
||||||
|
class:large={item.displayTileSize === 4}
|
||||||
|
class:startsNewGroup={item.startsNewGroup}
|
||||||
|
class:endsGroup={item.endsGroup}
|
||||||
|
data-index={index}
|
||||||
|
data-item={JSON.stringify(item)}
|
||||||
|
data-id={JSON.stringify(getKey(item))}
|
||||||
|
draggable="true"
|
||||||
|
on:dragstart={start}
|
||||||
|
on:dragover={over}
|
||||||
|
on:dragleave={leave}
|
||||||
|
on:drop={drop}
|
||||||
|
on:click={click}
|
||||||
|
in:receive={{ key: getKey(item) }}
|
||||||
|
out:send={{ key: getKey(item) }}
|
||||||
|
animate:flip={{ duration: 300 }}
|
||||||
|
class:over={getKey(item) === isOver}>
|
||||||
|
|
||||||
|
<slot {item} {index}>
|
||||||
|
<p>{getKey(item)}</p>
|
||||||
|
</slot>
|
||||||
|
|
||||||
|
</li>
|
||||||
|
{/each}
|
||||||
|
</ul>
|
||||||
|
{/if}
|
@ -1,6 +1,7 @@
|
|||||||
<script>
|
<script>
|
||||||
export let name = "%USER%";
|
export let name = "%USER%";
|
||||||
export let icon;
|
export let icon;
|
||||||
|
export let onClick;
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
@ -18,4 +19,4 @@
|
|||||||
</style>
|
</style>
|
||||||
|
|
||||||
|
|
||||||
<a href="404"> <img src="/user.svg"> </a>
|
<a on:click={onClick} href="404"> <img src="/user.svg"> </a>
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script>
|
<script>
|
||||||
import Nav from '../components/Nav.svelte';
|
import Nav from '../components/Nav.svelte';
|
||||||
import userManager from '../components/userManager.svelte'
|
import userManager from '../components/userManager.svelte'
|
||||||
|
import Modal from '../components/Modal.svelte'
|
||||||
|
|
||||||
export let segment;
|
export let segment;
|
||||||
</script>
|
</script>
|
||||||
@ -34,6 +35,7 @@
|
|||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
|
<Modal
|
||||||
<Nav {segment} rightComponent={userManager}/>
|
<Nav {segment} rightComponent={userManager}/>
|
||||||
|
|
||||||
<slot></slot>
|
<slot></slot>
|
||||||
|
6
static/css/bootstrap.min.css
vendored
Executable file
6
static/css/bootstrap.min.css
vendored
Executable file
File diff suppressed because one or more lines are too long
140
static/css/global.css
Executable file
140
static/css/global.css
Executable file
@ -0,0 +1,140 @@
|
|||||||
|
body {
|
||||||
|
font-family: 'Open Sans', sans-serif;
|
||||||
|
font-display: swap;
|
||||||
|
font-size: 1rem;
|
||||||
|
line-height: 1.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay {
|
||||||
|
display: flex;
|
||||||
|
background-size: cover;
|
||||||
|
background-position: center;
|
||||||
|
min-height: 200px;
|
||||||
|
text-align: center;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
display: inline-block;
|
||||||
|
font-size: 3rem;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav {
|
||||||
|
align-items: center;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* flexbox navbar test*/
|
||||||
|
nav ul {
|
||||||
|
display: flex;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav ul li {
|
||||||
|
font-size: 1rem;
|
||||||
|
text-align: center;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav li {
|
||||||
|
list-style-type: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Change color on hover */
|
||||||
|
nav li a {
|
||||||
|
margin-left: 20px;
|
||||||
|
margin-right: 20px;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
main {
|
||||||
|
margin-top: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Content types */
|
||||||
|
section {
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 5px 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*Simple css to style it like a toggle switch*/
|
||||||
|
.theme-switch-wrapper {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-switch {
|
||||||
|
display: inline-block;
|
||||||
|
height: 24px;
|
||||||
|
position: relative;
|
||||||
|
width: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-switch input {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.slider {
|
||||||
|
background-color: #ccc;
|
||||||
|
bottom: 0;
|
||||||
|
cursor: pointer;
|
||||||
|
left: 0;
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
top: 0;
|
||||||
|
transition: .2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.slider:before {
|
||||||
|
background-color: #fff;
|
||||||
|
bottom: 4px;
|
||||||
|
content: "";
|
||||||
|
height: 16px;
|
||||||
|
left: 4px;
|
||||||
|
position: absolute;
|
||||||
|
transition: .2s;
|
||||||
|
width: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
input:checked + .slider {
|
||||||
|
background-color: #66bb6a;
|
||||||
|
}
|
||||||
|
|
||||||
|
input:checked + .slider:before {
|
||||||
|
transform: translateX(16px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.slider.round {
|
||||||
|
border-radius: 34px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.slider.round:before {
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Responsive layout - when the screen is less than 700px wide, make the two columns stack on top of each other instead of next to each other */
|
||||||
|
@media screen and (max-width: 700px) {
|
||||||
|
.title {
|
||||||
|
display: block;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav {
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav ul {
|
||||||
|
flex-direction: column;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav ul li a {
|
||||||
|
text-align: center;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
}
|
43
static/css/themes.css
Executable file
43
static/css/themes.css
Executable file
@ -0,0 +1,43 @@
|
|||||||
|
/* Default theme */
|
||||||
|
:root {
|
||||||
|
--background-color: #222222;
|
||||||
|
--headings-color: #ffffff;
|
||||||
|
--font-color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Dark theme */
|
||||||
|
html[data-theme='dark'] {
|
||||||
|
--background-color: #222222;
|
||||||
|
--headings-color: #ffffff;
|
||||||
|
--font-color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: var(--background-color);
|
||||||
|
color: var(--font-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
h1, h2, h3, h4, h5 {
|
||||||
|
color: var(--headings-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
background-color: #eeeeee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay {
|
||||||
|
border-bottom: 2px solid #eeeeee;
|
||||||
|
}
|
||||||
|
|
||||||
|
li a:hover {
|
||||||
|
border-bottom: 3px solid white;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #CC99FF;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover {
|
||||||
|
color: white;
|
||||||
|
}
|
BIN
static/img/bluenebula.png
Executable file
BIN
static/img/bluenebula.png
Executable file
Binary file not shown.
After Width: | Height: | Size: 1.6 MiB |
BIN
static/img/bluenebula.webp
Normal file
BIN
static/img/bluenebula.webp
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.9 KiB |
BIN
static/img/favicon.ico
Executable file
BIN
static/img/favicon.ico
Executable file
Binary file not shown.
After Width: | Height: | Size: 15 KiB |
BIN
static/img/friendscats1.png
Executable file
BIN
static/img/friendscats1.png
Executable file
Binary file not shown.
After Width: | Height: | Size: 140 KiB |
5
static/img/user.svg
Normal file
5
static/img/user.svg
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="white"
|
||||||
|
stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="feather feather-user">
|
||||||
|
<path d="M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2"></path>
|
||||||
|
<circle cx="12" cy="7" r="4"></circle>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 322 B |
Loading…
x
Reference in New Issue
Block a user