2019-08-02 15:54:10 +02:00
|
|
|
<script>
|
|
|
|
|
|
|
|
import {
|
|
|
|
last,
|
|
|
|
sortBy,
|
|
|
|
filter,
|
|
|
|
map,
|
|
|
|
uniqWith,
|
|
|
|
isEqual,
|
|
|
|
trimCharsStart,
|
|
|
|
trimChars,
|
2019-08-15 09:49:15 +02:00
|
|
|
join,
|
|
|
|
includes
|
2019-08-02 15:54:10 +02:00
|
|
|
} from "lodash/fp";
|
|
|
|
|
|
|
|
import { pipe } from "../common/core";
|
|
|
|
import getIcon from "../common/icon";
|
|
|
|
import { store } from "../builderStore";
|
|
|
|
|
|
|
|
export let components = []
|
|
|
|
export let thisLevel = "";
|
|
|
|
|
2019-08-15 09:49:15 +02:00
|
|
|
let pathPartsThisLevel;
|
|
|
|
let componentsThisLevel;
|
|
|
|
let subfolders;
|
|
|
|
|
|
|
|
let expandedFolders = [];
|
|
|
|
|
2019-08-02 15:54:10 +02:00
|
|
|
const joinPath = join("/");
|
|
|
|
|
|
|
|
const normalizedName = name => pipe(name, [
|
|
|
|
trimCharsStart("./"),
|
|
|
|
trimCharsStart("~/"),
|
|
|
|
trimCharsStart("../"),
|
|
|
|
trimChars(" ")
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
const isOnThisLevel = (c) =>
|
|
|
|
normalizedName(c.name).split("/").length === pathPartsThisLevel
|
|
|
|
&&
|
|
|
|
(!thisLevel || normalizedName(c.name).startsWith(normalizedName(thisLevel)));
|
|
|
|
|
|
|
|
const notOnThisLevel = (c) => !isOnThisLevel(c);
|
|
|
|
|
|
|
|
const isInSubfolder = (subfolder, c) =>
|
|
|
|
normalizedName(c.name).startsWith(
|
|
|
|
trimCharsStart("/")(
|
|
|
|
joinPath([thisLevel, subfolder])));
|
|
|
|
|
|
|
|
const isOnNextLevel = (c) =>
|
|
|
|
normalizedName(c.name).split("/").length === pathPartsThisLevel + 1
|
|
|
|
|
|
|
|
const lastPartOfName = (c) =>
|
|
|
|
last(c.name.split("/"))
|
|
|
|
|
|
|
|
const subFolder = (c) => {
|
|
|
|
const cname = normalizedName(c.name);
|
|
|
|
const folderName = cname.substring(thisLevel.length, cname.length).split("/")[0];
|
|
|
|
|
|
|
|
return ({
|
|
|
|
name: folderName,
|
2019-08-15 09:49:15 +02:00
|
|
|
isExpanded: includes(folderName)(expandedFolders),
|
2019-08-02 15:54:10 +02:00
|
|
|
path: thisLevel + "/" + folderName
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const subComponents = (subfolder) => pipe(components, [
|
|
|
|
filter(c => isInSubfolder(subfolder, c))
|
|
|
|
]);
|
|
|
|
|
|
|
|
const expandFolder = folder => {
|
|
|
|
const expandedFolder = {...folder};
|
2019-08-15 09:49:15 +02:00
|
|
|
if(expandedFolder.isExpanded) {
|
|
|
|
expandedFolder.isExpanded = false;
|
|
|
|
expandedFolders = filter(f => f.name !== folder.name)(expandedFolders);
|
|
|
|
} else {
|
|
|
|
expandedFolder.isExpanded = true;
|
|
|
|
expandedFolders.push(folder.name);
|
|
|
|
}
|
2019-08-02 15:54:10 +02:00
|
|
|
const newFolders = [...subfolders];
|
|
|
|
newFolders.splice(
|
|
|
|
newFolders.indexOf(folder),
|
|
|
|
1,
|
|
|
|
expandedFolder);
|
|
|
|
subfolders = newFolders;
|
2019-08-15 09:49:15 +02:00
|
|
|
|
2019-08-02 15:54:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const isComponentSelected = (current,c) =>
|
|
|
|
current
|
|
|
|
&& current.name === c.name
|
|
|
|
|
|
|
|
const isFolderSelected = (current, folder) =>
|
|
|
|
isInSubfolder(current, folder)
|
|
|
|
|
2019-08-15 09:49:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
$: {
|
|
|
|
pathPartsThisLevel = !thisLevel
|
|
|
|
? 1
|
|
|
|
: normalizedName(thisLevel).split("/").length + 1;
|
|
|
|
|
|
|
|
componentsThisLevel =
|
|
|
|
pipe(components, [
|
|
|
|
filter(isOnThisLevel),
|
|
|
|
map(c => ({component:c, title:lastPartOfName(c)})),
|
|
|
|
sortBy("title")
|
|
|
|
]);
|
|
|
|
|
|
|
|
subfolders =
|
|
|
|
pipe(components, [
|
|
|
|
filter(notOnThisLevel),
|
|
|
|
sortBy("name"),
|
|
|
|
map(subFolder),
|
|
|
|
uniqWith((f1,f2) => f1.path === f2.path)
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2019-08-02 15:54:10 +02:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<div class="root" style={`padding-left: calc(10px * ${pathPartsThisLevel})`}>
|
|
|
|
|
|
|
|
{#each subfolders as folder}
|
|
|
|
<div class="hierarchy-item folder"
|
|
|
|
on:click|stopPropagation={() => expandFolder(folder)}>
|
|
|
|
<span>{@html getIcon(folder.isExpanded ? "chevron-down" : "chevron-right", "16")}</span>
|
|
|
|
<span class="title" class:currentfolder={$store.currentFrontEndItem && isInSubfolder(folder.name, $store.currentFrontEndItem)}>{folder.name}</span>
|
|
|
|
{#if folder.isExpanded}
|
|
|
|
<svelte:self components={subComponents(folder.name)}
|
|
|
|
thisLevel={folder.path} />
|
|
|
|
{/if}
|
|
|
|
</div>
|
|
|
|
{/each}
|
|
|
|
|
|
|
|
{#each componentsThisLevel as component}
|
|
|
|
<div class="hierarchy-item component" class:selected={isComponentSelected($store.currentFrontEndItem, component.component)}
|
|
|
|
on:click|stopPropagation={() => store.setCurrentComponent(component.component)}>
|
|
|
|
<span>{@html getIcon("circle", "7")}</span>
|
|
|
|
<span class="title">{component.title}</span>
|
|
|
|
</div>
|
|
|
|
{/each}
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<style>
|
|
|
|
|
|
|
|
.root {
|
|
|
|
color: var(--secondary50);
|
2019-09-23 23:22:57 +02:00
|
|
|
font-size: .9rem;
|
2019-09-25 21:53:52 +02:00
|
|
|
font-weight: bold;
|
2019-08-02 15:54:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
.hierarchy-item {
|
|
|
|
cursor: pointer;
|
2019-08-04 23:21:16 +02:00
|
|
|
padding: 5px 0px;
|
2019-08-02 15:54:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
.hierarchy-item:hover {
|
2019-09-23 23:22:57 +02:00
|
|
|
color: var(--secondary);
|
2019-08-02 15:54:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
.component {
|
|
|
|
margin-left: 5px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.currentfolder {
|
|
|
|
color: var(--secondary100);
|
|
|
|
}
|
|
|
|
|
|
|
|
.selected {
|
|
|
|
color: var(--primary100);
|
2019-09-23 23:22:57 +02:00
|
|
|
font-weight: bold;
|
2019-08-02 15:54:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
.title {
|
|
|
|
margin-left: 10px;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
</style>
|