Gisle Aune
3 years ago
29 changed files with 524 additions and 64 deletions
-
4api/project.go
-
2database/postgres/projectgroups.go
-
5models/project.go
-
2services/loader.go
-
12svelte-ui/src/App.svelte
-
17svelte-ui/src/clients/stufflog.ts
-
6svelte-ui/src/components/ChildEntry.svelte
-
17svelte-ui/src/components/ColoredNumber.svelte
-
4svelte-ui/src/components/FocusHandler.svelte
-
3svelte-ui/src/components/Modal.svelte
-
11svelte-ui/src/components/OptionRow.svelte
-
111svelte-ui/src/components/ProjectGroupMenu.svelte
-
19svelte-ui/src/components/ProjectGroupSelect.svelte
-
1svelte-ui/src/components/ProjectSelect.svelte
-
3svelte-ui/src/components/QLList.svelte
-
10svelte-ui/src/components/QLListItem.svelte
-
84svelte-ui/src/components/QuestLog.svelte
-
2svelte-ui/src/components/StatusColor.svelte
-
2svelte-ui/src/components/Tag.svelte
-
19svelte-ui/src/forms/ProjectForm.svelte
-
105svelte-ui/src/forms/ProjectGroupForm.svelte
-
3svelte-ui/src/models/project.ts
-
18svelte-ui/src/models/projectgroup.ts
-
2svelte-ui/src/pages/ProjectPage.svelte
-
36svelte-ui/src/pages/QLPage.svelte
-
2svelte-ui/src/stores/markStale.ts
-
6svelte-ui/src/stores/modal.ts
-
48svelte-ui/src/stores/projectGroup.ts
-
28svelte-ui/src/utils/sorters.ts
@ -0,0 +1,17 @@ |
|||||
|
<script lang="ts"> |
||||
|
import StatusColor from "./StatusColor.svelte"; |
||||
|
|
||||
|
export let selected: boolean = false; |
||||
|
export let status: string = "to do"; |
||||
|
export let number: number | null | undefined = void(0); |
||||
|
|
||||
|
let entry: {statusTag: string} |
||||
|
|
||||
|
$: entry = {statusTag: status}; |
||||
|
</script> |
||||
|
|
||||
|
{#if (!!number)} |
||||
|
<StatusColor selected={selected} affects="project" entry={entry}> |
||||
|
<div class="sccfg">{number}</div> |
||||
|
</StatusColor> |
||||
|
{/if} |
@ -0,0 +1,111 @@ |
|||||
|
<script lang="ts"> |
||||
|
import { tick } from "svelte"; |
||||
|
import { navigate } from "svelte-routing"; |
||||
|
|
||||
|
import type { ProjectGroupResult } from "../models/projectgroup"; |
||||
|
import modalStore from "../stores/modal"; |
||||
|
import projectGroupStore from "../stores/projectGroup"; |
||||
|
import { sortProjects } from "../utils/sorters"; |
||||
|
import ColoredNumber from "./ColoredNumber.svelte"; |
||||
|
|
||||
|
export let groups: ProjectGroupResult[] = []; |
||||
|
export let selected: string; |
||||
|
|
||||
|
let failedCount = 0; |
||||
|
|
||||
|
function onClickAdd() { |
||||
|
modalStore.set({name: "projectgroup.add"}); |
||||
|
} |
||||
|
|
||||
|
function onNavigate(group: ProjectGroupResult) { |
||||
|
if (group.projects.length === 0) { |
||||
|
navigate(`/questlog/${group.id}`); |
||||
|
} else { |
||||
|
const projetcs = [...group.projects].sort(sortProjects); |
||||
|
|
||||
|
navigate(`/questlog/${group.id}/${projetcs[0].id}`); |
||||
|
|
||||
|
// There's some weirdness with navigate. This hack will just do a |
||||
|
// groups = [...groups] in the sttore to get the page to update. |
||||
|
tick().then(() => { |
||||
|
projectGroupStore.fakeRefresh(); |
||||
|
}) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
</script> |
||||
|
|
||||
|
<div class="group-menu"> |
||||
|
{#each groups as group (group.id)} |
||||
|
<div class="group-entry" on:click={() => onNavigate(group)} class:selected={selected === group.id}> |
||||
|
<div class="name">{group.abbreviation}</div> |
||||
|
<div class="counts"> |
||||
|
<ColoredNumber selected={selected === group.id} status="active" number={group.projectCounts["active"]} /> |
||||
|
<ColoredNumber selected={selected === group.id} status="background" number={group.projectCounts["background"]} /> |
||||
|
<ColoredNumber selected={selected === group.id} status="progress" number={group.projectCounts["progress"]} /> |
||||
|
<ColoredNumber selected={selected === group.id} status="to do" number={group.projectCounts["to do"]} /> |
||||
|
<ColoredNumber selected={selected === group.id} status="on hold" number={group.projectCounts["on hold"]} /> |
||||
|
</div> |
||||
|
</div> |
||||
|
{/each} |
||||
|
|
||||
|
<div class="group-entry add" on:click={onClickAdd}> |
||||
|
<div>+</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
|
||||
|
<style> |
||||
|
div.group-menu { |
||||
|
display: flex; |
||||
|
flex-direction: row; |
||||
|
padding: 0 2ch; |
||||
|
border-bottom: 1px solid #333; |
||||
|
margin-bottom: 0.5em; |
||||
|
|
||||
|
-webkit-user-select: none; |
||||
|
-moz-user-select: none; |
||||
|
} |
||||
|
|
||||
|
div.group-entry { |
||||
|
padding: 0.05em 1ch; |
||||
|
padding-bottom: 0.15em; |
||||
|
margin-right: 2ch; |
||||
|
border: 1px solid #333; |
||||
|
border-bottom: none; |
||||
|
border-top-left-radius: 0.5em; |
||||
|
border-top-right-radius: 0.5em; |
||||
|
cursor: pointer; |
||||
|
color: #777; |
||||
|
font-weight: 300; |
||||
|
} |
||||
|
div.group-entry.add { |
||||
|
color: #333; |
||||
|
} |
||||
|
div.group-entry.add > div { |
||||
|
font-size: 2em; |
||||
|
line-height: 1em; |
||||
|
} |
||||
|
div.group-entry.selected { |
||||
|
background-color: #191919; |
||||
|
color: #AAA; |
||||
|
} |
||||
|
div.group-entry:hover { |
||||
|
color: #CCC; |
||||
|
background-color: #222222; |
||||
|
} |
||||
|
|
||||
|
div.name { |
||||
|
text-align: center; |
||||
|
padding: 0em 0.4ch; |
||||
|
} |
||||
|
|
||||
|
div.counts { |
||||
|
font-size: 0.666em; |
||||
|
font-weight: 400; |
||||
|
text-align: center; |
||||
|
} |
||||
|
div.counts :global(div) { |
||||
|
display: inline-block; |
||||
|
padding: 0em 0.2ch; |
||||
|
} |
||||
|
</style> |
@ -0,0 +1,19 @@ |
|||||
|
<script lang="ts"> |
||||
|
import projectGroupStore from "../stores/projectGroup"; |
||||
|
|
||||
|
export let value = ""; |
||||
|
export let name = ""; |
||||
|
export let disabled = false; |
||||
|
export let optional = false; |
||||
|
</script> |
||||
|
|
||||
|
<select name={name} bind:value={value} disabled={disabled || $projectGroupStore.loading}> |
||||
|
{#if optional} |
||||
|
<option value={""} selected={"" === value}>None</option> |
||||
|
{/if} |
||||
|
{#each $projectGroupStore.groups as group (group.id)} |
||||
|
{#if group.id !== "META_UNGROUPED"} |
||||
|
<option value={group.id} selected={group.id === value}>{group.name}</option> |
||||
|
{/if} |
||||
|
{/each} |
||||
|
</select> |
@ -0,0 +1,105 @@ |
|||||
|
<script lang="ts"> |
||||
|
import { navigate } from "svelte-routing"; |
||||
|
|
||||
|
import stuffLogClient from "../clients/stufflog"; |
||||
|
import Modal from "../components/Modal.svelte"; |
||||
|
import type ProjectGroup from "../models/projectgroup"; |
||||
|
import markStale from "../stores/markStale"; |
||||
|
import modalStore from "../stores/modal"; |
||||
|
|
||||
|
export let deletion = false; |
||||
|
export let creation = false; |
||||
|
|
||||
|
const md = $modalStore; |
||||
|
let group: ProjectGroup = { |
||||
|
id: "", |
||||
|
name: "", |
||||
|
description: "", |
||||
|
abbreviation: "", |
||||
|
categoryNames: {}, |
||||
|
} |
||||
|
let verb = "Add"; |
||||
|
if (md.name === "projectgroup.edit" || md.name === "projectgroup.delete") { |
||||
|
group = md.projectGroup; |
||||
|
verb = (md.name === "projectgroup.edit") ? "Edit" : "Delete"; |
||||
|
} else if (md.name !== "projectgroup.add") { |
||||
|
throw new Error("Wrong form") |
||||
|
} |
||||
|
|
||||
|
let name = group.name; |
||||
|
let description = group.description; |
||||
|
let abbreviation = group.abbreviation; |
||||
|
let categoryNames = group.categoryNames || {}; |
||||
|
let error = null; |
||||
|
let loading = false; |
||||
|
|
||||
|
function onSubmit() { |
||||
|
loading = true; |
||||
|
error = null; |
||||
|
|
||||
|
if (creation) { |
||||
|
stuffLogClient.createProjectGroup({ |
||||
|
name, abbreviation, description, categoryNames |
||||
|
}).then(newGroup => { |
||||
|
markStale("project"); |
||||
|
modalStore.close(); |
||||
|
navigate(`/questlog/${newGroup.id}`); |
||||
|
}).catch(err => { |
||||
|
error = err.message ? err.message : err.toString(); |
||||
|
}).finally(() => { |
||||
|
loading = false; |
||||
|
}) |
||||
|
} else if (deletion) { |
||||
|
stuffLogClient.deleteProjectGroup(group.id).then(() => { |
||||
|
markStale("project"); |
||||
|
modalStore.close(); |
||||
|
navigate("/questlog/"); |
||||
|
}).catch(err => { |
||||
|
error = err.message ? err.message : err.toString(); |
||||
|
}).finally(() => { |
||||
|
loading = false; |
||||
|
}) |
||||
|
} else { |
||||
|
stuffLogClient.updateProjectGroup(group.id, { |
||||
|
name, abbreviation, description, |
||||
|
setCategoryNames: categoryNames, |
||||
|
}).then(() => { |
||||
|
markStale("project"); |
||||
|
modalStore.close(); |
||||
|
}).catch(err => { |
||||
|
error = err.message ? err.message : err.toString(); |
||||
|
}).finally(() => { |
||||
|
loading = false; |
||||
|
}) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
function onClose() { |
||||
|
modalStore.close(); |
||||
|
} |
||||
|
</script> |
||||
|
|
||||
|
<Modal show title="{verb} Project Group" error={error} closable on:close={onClose}> |
||||
|
<form on:submit|preventDefault={onSubmit}> |
||||
|
<label for="name">Name</label> |
||||
|
<input disabled={deletion} name="name" type="text" bind:value={name} /> |
||||
|
<label for="abbreviation">Abbreviation</label> |
||||
|
<input disabled={deletion} name="abbreviation" bind:value={abbreviation} /> |
||||
|
<label for="description">Description</label> |
||||
|
<textarea disabled={deletion} name="description" bind:value={description} /> |
||||
|
|
||||
|
<label for="name">Custom Labels</label> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="Deadlines" bind:value={categoryNames["deadlines"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="Active" bind:value={categoryNames["active"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="Background" bind:value={categoryNames["background"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="Progress" bind:value={categoryNames["progress"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="To Do" bind:value={categoryNames["to do"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="On Hold" bind:value={categoryNames["on hold"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="Completed" bind:value={categoryNames["completed"]} /> |
||||
|
<input disabled={deletion} name="name" type="text" placeholder="Failed" bind:value={categoryNames["failed"]} /> |
||||
|
|
||||
|
<hr /> |
||||
|
|
||||
|
<button disabled={loading} type="submit">{verb} Project Group</button> |
||||
|
</form> |
||||
|
</Modal> |
@ -0,0 +1,48 @@ |
|||||
|
import { writable } from "svelte/store"; |
||||
|
import stuffLogClient from "../clients/stufflog"; |
||||
|
import type { ProjectGroupResult } from "../models/projectgroup"; |
||||
|
|
||||
|
interface ProjectStoreData { |
||||
|
loading: boolean |
||||
|
stale: boolean |
||||
|
groups: ProjectGroupResult[] |
||||
|
} |
||||
|
|
||||
|
function createProjectGroupStore() { |
||||
|
const {update, subscribe} = writable<ProjectStoreData>({ |
||||
|
loading: false, |
||||
|
stale: true, |
||||
|
groups: [], |
||||
|
}); |
||||
|
|
||||
|
return { |
||||
|
subscribe, |
||||
|
|
||||
|
markStale() { |
||||
|
update(v => ({...v, stale: true})); |
||||
|
}, |
||||
|
|
||||
|
async loadOne(id: string) { |
||||
|
update(v => ({...v, loading: true})); |
||||
|
const group = await stuffLogClient.findProjectGroup(id); |
||||
|
update(v => ({...v, loading: false, groups: [ |
||||
|
...v.groups.filter(g => g.id === id), |
||||
|
group, |
||||
|
]})); |
||||
|
}, |
||||
|
|
||||
|
fakeRefresh() { |
||||
|
update(v => ({...v, groups: [...v.groups]})) |
||||
|
}, |
||||
|
|
||||
|
async load() { |
||||
|
update(v => ({...v, loading: true, stale: false})); |
||||
|
const groups = await stuffLogClient.listProjectGroups(); |
||||
|
update(v => ({...v, loading: false, groups: groups})); |
||||
|
}, |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
const projectGroupStore = createProjectGroupStore(); |
||||
|
|
||||
|
export default projectGroupStore; |
@ -0,0 +1,28 @@ |
|||||
|
import type { ProjectResult } from "../models/project"; |
||||
|
|
||||
|
const STATUS_ORDER: (string | undefined | null)[] = [ |
||||
|
"deadlines", |
||||
|
null, |
||||
|
void(0), |
||||
|
"active", |
||||
|
"background", |
||||
|
"progress", |
||||
|
"to do", |
||||
|
"on hold", |
||||
|
"completed", |
||||
|
"failed", |
||||
|
"declined", |
||||
|
] |
||||
|
|
||||
|
export function sortProjects(a: ProjectResult, b: ProjectResult) { |
||||
|
const as = STATUS_ORDER.indexOf(a.statusTag); |
||||
|
const bs = STATUS_ORDER.indexOf(b.statusTag); |
||||
|
if (as !== bs) { |
||||
|
return as - bs; |
||||
|
} |
||||
|
|
||||
|
const aName = `${a.tags.slice(0, 1).map(t => t+":").join("")} ${a.name}`.trim(); |
||||
|
const bName = `${b.tags.slice(0, 1).map(t => t+":").join("")} ${b.name}`.trim(); |
||||
|
|
||||
|
return aName.localeCompare(bName); |
||||
|
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue