2024-02-15 18:52:46 +01:00
|
|
|
<script context="module">
|
|
|
|
import parse from "pure-color/parse"
|
|
|
|
import rgb2hsl from "pure-color/convert/rgb2hsl";
|
|
|
|
import hsl2rgb from "pure-color/convert/hsl2rgb";
|
|
|
|
import rgb2hex from "pure-color/convert/rgb2hex";
|
|
|
|
|
|
|
|
// Generate a branding style from a file's properties map
|
|
|
|
export const branding_from_path = path => {
|
|
|
|
let style = {}
|
|
|
|
for (let node of path) {
|
|
|
|
add_styles(style, node.properties)
|
|
|
|
}
|
|
|
|
last_generated_style = style
|
|
|
|
return gen_css(style)
|
|
|
|
}
|
|
|
|
|
|
|
|
// The last style which was generated is cached, when we don't have a complete
|
|
|
|
// path to generate the style with we will use the cached style as a basis
|
|
|
|
let last_generated_style = {}
|
|
|
|
export const branding_from_node = node => {
|
|
|
|
add_styles(last_generated_style, node.properties)
|
|
|
|
return gen_css(last_generated_style)
|
|
|
|
}
|
|
|
|
|
|
|
|
const gen_css = style => {
|
|
|
|
return Object.entries(style).map(([key, value]) => `--${key}:${value}`).join(';');
|
|
|
|
}
|
|
|
|
|
|
|
|
// add_styles adds the styles configured in the properties struct to the
|
|
|
|
// existing style which is passed as the first argument. When navigating to a
|
|
|
|
// path this function is executed on every member of the path so all the styles
|
|
|
|
// get combined
|
|
|
|
const add_styles = (style, properties) => {
|
|
|
|
if (!properties || !properties.branding_enabled || properties.branding_enabled !== "true") {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if (properties.brand_input_color) {
|
|
|
|
style.input_background = properties.brand_input_color
|
|
|
|
style.input_hover_background = properties.brand_input_color
|
|
|
|
style.input_text = add_light(properties.brand_input_color, 70)
|
|
|
|
}
|
|
|
|
if (properties.brand_highlight_color) {
|
|
|
|
style.highlight_color = properties.brand_highlight_color
|
|
|
|
style.highlight_background = properties.brand_highlight_color
|
|
|
|
style.highlight_text_color = add_light(properties.brand_highlight_color, 70)
|
2024-02-16 11:49:38 +01:00
|
|
|
style.link_color = properties.brand_highlight_color
|
2024-02-15 18:52:46 +01:00
|
|
|
}
|
|
|
|
if (properties.brand_danger_color) {
|
|
|
|
style.danger_color = properties.brand_danger_color
|
|
|
|
style.danger_text_color = add_light(properties.brand_danger_color, 70)
|
|
|
|
}
|
|
|
|
if (properties.brand_background_color) {
|
|
|
|
style.background_color = properties.brand_background_color
|
|
|
|
style.background = properties.brand_background_color
|
|
|
|
style.background_text_color = add_light(properties.brand_background_color, 70)
|
2024-02-16 11:49:38 +01:00
|
|
|
style.background_pattern_color = properties.brand_background_color
|
2024-02-15 18:52:46 +01:00
|
|
|
}
|
|
|
|
if (properties.brand_body_color) {
|
|
|
|
style.body_color = properties.brand_body_color
|
|
|
|
style.body_background = properties.brand_body_color
|
|
|
|
style.body_text_color = add_light(properties.brand_body_color, 70)
|
|
|
|
style.shaded_background = set_alpha(properties.brand_body_color, 0.8)
|
|
|
|
style.separator = add_light(properties.brand_body_color, 5)
|
2024-02-16 11:49:38 +01:00
|
|
|
style.shadow_color = darken(properties.brand_body_color, 0.8)
|
2024-02-15 18:52:46 +01:00
|
|
|
}
|
|
|
|
if (properties.brand_card_color) {
|
|
|
|
style.card_color = properties.brand_card_color
|
|
|
|
}
|
|
|
|
if (properties.brand_background_image) {
|
|
|
|
style.background_image = "url('/api/filesystem/"+properties.brand_background_image+"')"
|
|
|
|
style.background_image_size = "cover"
|
|
|
|
style.background_image_position = "center"
|
|
|
|
style.background_image_repeat = "no-repeat"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const add_light = (color, amt) => {
|
|
|
|
let hsl = rgb2hsl(parse(color)) // Convert hex to hsl
|
2024-02-16 11:49:38 +01:00
|
|
|
// If the lightness is less than 30 it is considered a dark colour. This
|
|
|
|
// threshold is 30 instead of 50 because overall dark text is more legible
|
|
|
|
if (hsl[2] < 30) {
|
2024-02-15 18:52:46 +01:00
|
|
|
hsl[2] = hsl[2]+amt // Dark color, add lightness
|
|
|
|
} else {
|
|
|
|
hsl[2] = hsl[2]-amt // Light color, remove lightness
|
|
|
|
}
|
|
|
|
return rgb2hex(hsl2rgb(hsl)) // Convert back to hex
|
|
|
|
}
|
2024-02-16 11:49:38 +01:00
|
|
|
|
|
|
|
// Darken and desaturate. Only used for shadows
|
|
|
|
const darken = (color, percent) => {
|
|
|
|
let hsl = rgb2hsl(parse(color)) // Convert hex to hsl
|
|
|
|
hsl[1] = hsl[1]*percent
|
|
|
|
hsl[2] = hsl[2]*percent
|
|
|
|
return rgb2hex(hsl2rgb(hsl)) // Convert back to hex
|
|
|
|
}
|
|
|
|
|
2024-02-15 18:52:46 +01:00
|
|
|
const set_alpha = (color, amt) => {
|
|
|
|
let rgb = parse(color)
|
|
|
|
rgb.push(amt)
|
|
|
|
return "rgba("+rgb.join(", ")+")"
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import { createEventDispatcher } from "svelte";
|
|
|
|
import FilePicker from "./filemanager/FilePicker.svelte";
|
|
|
|
import { fs_update } from "./FilesystemAPI";
|
|
|
|
import { fs_node_type } from "./FilesystemUtil";
|
|
|
|
|
|
|
|
let dispatch = createEventDispatcher()
|
|
|
|
|
|
|
|
export let file = {
|
|
|
|
properties: {
|
|
|
|
branding_enabled: "false",
|
|
|
|
brand_input_color: "#2d2d2d",
|
|
|
|
brand_highlight_color: "#75b72d",
|
|
|
|
brand_danger_color: "#bd5f69",
|
|
|
|
brand_background_color: "#141414",
|
|
|
|
brand_body_color: "#1e1e1e",
|
|
|
|
brand_card_color: "#282828",
|
|
|
|
brand_header_image: "",
|
|
|
|
brand_header_link: "",
|
|
|
|
brand_footer_image: "",
|
|
|
|
brand_footer_link: "",
|
|
|
|
brand_background_image: "",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
export let enabled = false
|
|
|
|
|
|
|
|
$: update_colors(file)
|
|
|
|
const update_colors = file => {
|
|
|
|
if (enabled) {
|
|
|
|
file.properties.branding_enabled = "true"
|
|
|
|
dispatch("style_change", file)
|
|
|
|
} else {
|
|
|
|
file.properties.branding_enabled = ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let picker
|
|
|
|
let picking = ""
|
|
|
|
const pick_image = type => {
|
|
|
|
picking = type
|
|
|
|
picker.open(file.path)
|
|
|
|
}
|
|
|
|
const handle_picker = async e => {
|
|
|
|
if (e.detail.length !== 1) {
|
|
|
|
alert("Please select one file")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
let f = e.detail[0]
|
|
|
|
let file_id = f.id
|
|
|
|
|
|
|
|
if (fs_node_type(f) !== "image") {
|
|
|
|
alert("Please select an image file")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this image is not public, it will be made public
|
|
|
|
if (file_id === undefined || file_id === "") {
|
|
|
|
try {
|
|
|
|
let new_file = await fs_update(e.detail[0].path, {shared: true})
|
|
|
|
file_id = new_file.id
|
|
|
|
} catch (err) {
|
|
|
|
alert(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (picking === "brand_header_image") {
|
|
|
|
file.properties.brand_header_image = file_id
|
|
|
|
} else if (picking === "brand_background_image") {
|
|
|
|
file.properties.brand_background_image = file_id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<p>
|
|
|
|
You can customize how your filesystem looks. The colours chosen here apply
|
|
|
|
to the directory you're currently in and all files and directories in this
|
|
|
|
directory. Colours which you do not want to modify can be left empty. Then
|
|
|
|
the default theme colour will be used.
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<div>
|
|
|
|
<input bind:checked={enabled} id="enable_branding" type="checkbox" class="form_input"/>
|
|
|
|
<label for="enable_branding">Enable custom branding options</label>
|
|
|
|
</div>
|
|
|
|
<hr/>
|
|
|
|
|
|
|
|
<div class="grid" class:disabled={!enabled}>
|
|
|
|
<div>Button</div>
|
|
|
|
<input type="color" bind:value={file.properties.brand_input_color}/>
|
|
|
|
<input type="text" bind:value={file.properties.brand_input_color}/>
|
|
|
|
<div>Highlighted button</div>
|
|
|
|
<input type="color" bind:value={file.properties.brand_highlight_color}/>
|
|
|
|
<input type="text" bind:value={file.properties.brand_highlight_color}/>
|
|
|
|
<div>Danger button</div>
|
|
|
|
<input type="color" bind:value={file.properties.brand_danger_color}/>
|
|
|
|
<input type="text" bind:value={file.properties.brand_danger_color}/>
|
|
|
|
<div>Background</div>
|
|
|
|
<input type="color" bind:value={file.properties.brand_background_color}/>
|
|
|
|
<input type="text" bind:value={file.properties.brand_background_color}/>
|
|
|
|
<div>Body</div>
|
|
|
|
<input type="color" bind:value={file.properties.brand_body_color}/>
|
|
|
|
<input type="text" bind:value={file.properties.brand_body_color}/>
|
|
|
|
<div>Card</div>
|
|
|
|
<input type="color" bind:value={file.properties.brand_card_color}/>
|
|
|
|
<input type="text" bind:value={file.properties.brand_card_color}/>
|
|
|
|
<div class="span3">
|
|
|
|
<hr/>
|
|
|
|
|
|
|
|
You can choose an image to show above or behind the files in this
|
2024-02-16 11:49:38 +01:00
|
|
|
directory. The image will be picked from your filesystem. The image will
|
|
|
|
get a shared file link. You can move and rename the file like normal,
|
|
|
|
but if you remove the shared file property your branding will stop
|
|
|
|
working.
|
2024-02-15 18:52:46 +01:00
|
|
|
</div>
|
|
|
|
<div>Header image ID</div>
|
|
|
|
<button on:click={() => pick_image("brand_header_image")}>
|
|
|
|
<i class="icon">folder_open</i>
|
|
|
|
Pick
|
|
|
|
</button>
|
|
|
|
<input type="text" bind:value={file.properties.brand_header_image}/>
|
|
|
|
<div>Header image link</div>
|
|
|
|
<input class="span2" type="text" bind:value={file.properties.brand_header_link}/>
|
|
|
|
<div>Background image ID</div>
|
|
|
|
<button on:click={() => pick_image("brand_background_image")}>
|
|
|
|
<i class="icon">folder_open</i>
|
|
|
|
Pick
|
|
|
|
</button>
|
|
|
|
<input type="text" bind:value={file.properties.brand_background_image}/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<hr/>
|
|
|
|
<p>
|
|
|
|
Below is an example of what the site looks like with these colours:
|
|
|
|
</p>
|
|
|
|
|
|
|
|
<div class="example example_background">
|
|
|
|
<div>The background</div>
|
|
|
|
|
|
|
|
<div class="example example_body">
|
|
|
|
<div>The content body. <a href="/">A link</a>!</div>
|
|
|
|
|
|
|
|
<hr/>
|
|
|
|
|
|
|
|
<div class="example_button_row">
|
|
|
|
<button type="button"><i class="icon">touch_app</i>Normal</button>
|
|
|
|
<button type="button" class="button_highlight"><i class="icon">priority_high</i>Important</button>
|
|
|
|
<button type="button" class="button_red"><i class="icon">warning</i>Dangerous</button>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="example_button_row">
|
|
|
|
<input type="text" value="A text field"/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<hr/>
|
|
|
|
|
|
|
|
<div class="example example_card">
|
|
|
|
<div>The top layer, for highlighted things</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<FilePicker bind:this={picker} on:files={handle_picker}/>
|
|
|
|
|
|
|
|
<style>
|
|
|
|
input[type="color"] {
|
|
|
|
padding: 0;
|
|
|
|
display: inline;
|
|
|
|
}
|
|
|
|
.grid {
|
|
|
|
display: grid;
|
|
|
|
width: 100%;
|
|
|
|
grid-template-columns: auto auto auto;
|
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
.span2 {
|
|
|
|
grid-column: span 2;
|
|
|
|
}
|
|
|
|
.span3 {
|
|
|
|
grid-column: span 3;
|
|
|
|
}
|
|
|
|
.disabled {
|
|
|
|
filter: brightness(50%);
|
|
|
|
}
|
|
|
|
|
|
|
|
.example {
|
|
|
|
margin: 6px 0;
|
|
|
|
padding: 10px;
|
|
|
|
border-radius: 10px;
|
|
|
|
}
|
|
|
|
.example_background {
|
|
|
|
background: var(--background);
|
|
|
|
color: var(--background_text_color)
|
|
|
|
}
|
|
|
|
.example_body {
|
|
|
|
background: var(--body_background);
|
|
|
|
color: var(--body_text_color)
|
|
|
|
}
|
|
|
|
.example_button_row {
|
|
|
|
display: flex;
|
|
|
|
flex-direction: row;
|
|
|
|
}
|
|
|
|
.example_button_row>* {
|
|
|
|
flex: 1 1 auto;
|
|
|
|
}
|
|
|
|
.example_card {
|
|
|
|
background: var(--card_color);
|
|
|
|
}
|
|
|
|
</style>
|