Add achievement version and commission filter

This commit is contained in:
Made Baruna 2021-09-08 19:36:14 +07:00
parent 351fd67b9e
commit a3259bbefa
No known key found for this signature in database
GPG key ID: 5AA5DA16AA5DCEAD
14 changed files with 142 additions and 63 deletions

View file

@ -5,7 +5,7 @@
import Icon from './Icon.svelte'; import Icon from './Icon.svelte';
const dispatch = createEventDispatcher(); const dispatch = createEventDispatcher();
export let className = ''; export let className = '';
export let icon = null; export let icon = null;
export let options; export let options;
@ -38,6 +38,7 @@
} }
selected = [...selectedMulti].map((e) => options[e]); selected = [...selectedMulti].map((e) => options[e]);
dispatch('change');
return; return;
} }
@ -101,19 +102,13 @@
$: iconClasses = focused ? 'transform rotate-180' : ''; $: iconClasses = focused ? 'transform rotate-180' : '';
</script> </script>
<style>
.hovered {
@apply text-white !important;
@apply bg-primary;
}
</style>
<svelte:window on:click={onWindowClick} on:keydown={onKeyDown} /> <svelte:window on:click={onWindowClick} on:keydown={onKeyDown} />
<div class={`select-none relative ${className}`} bind:this={container}> <div class={`select-none relative ${className}`} bind:this={container}>
<button <button
class={`flex w-full relative items-center px-4 bg-background rounded-2xl h-14 focus:outline-none focus:border-primary border-2 border-transparent ease-in duration-100 ${classes}`} class={`flex w-full relative items-center px-4 bg-background rounded-2xl h-14 focus:outline-none focus:border-primary border-2 border-transparent ease-in duration-100 ${classes}`}
on:click={toggleOptions}> on:click={toggleOptions}
>
{#if icon} {#if icon}
<Icon path={icon} color="white" className="mr-3" /> <Icon path={icon} color="white" className="mr-3" />
{/if} {/if}
@ -123,14 +118,16 @@
{#if focused} {#if focused}
<div <div
transition:fade={{ duration: 100 }} transition:fade={{ duration: 100 }}
class="bg-item rounded-2xl absolute mt-2 p-2 w-full z-50 flex flex-col text-white shadow-xl border border-background"> class="bg-item rounded-2xl absolute mt-2 p-2 w-full z-50 flex flex-col text-white shadow-xl border border-background"
>
{#each options as option, index} {#each options as option, index}
<span <span
on:click={() => !option.disabled && select(index)} on:click={() => !option.disabled && select(index)}
on:mouseenter={() => !option.disabled && onHover(index)} on:mouseenter={() => !option.disabled && onHover(index)}
class={`p-3 rounded-xl cursor-pointer flex class={`p-3 rounded-xl cursor-pointer flex
${selectedIndex === index || selectedMulti.has(index) ? 'text-primary font-semibold' : ''} ${selectedIndex === index || selectedMulti.has(index) ? 'text-primary font-semibold' : ''}
${hoveredIndex === index ? 'hovered' : ''}`}> ${hoveredIndex === index ? 'hovered' : ''}`}
>
{#if image}<img class="w-6 h-6 mr-2" src={option.image} alt={option.label} />{/if} {#if image}<img class="w-6 h-6 mr-2" src={option.image} alt={option.label} />{/if}
<span class="flex-1">{option.label}</span> <span class="flex-1">{option.label}</span>
{#if multiselect && selectedMulti.has(index)} {#if multiselect && selectedMulti.has(index)}
@ -141,3 +138,10 @@
</div> </div>
{/if} {/if}
</div> </div>
<style>
.hovered {
@apply text-white !important;
@apply bg-primary;
}
</style>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View file

@ -665,7 +665,11 @@
"title": "Achievement", "title": "Achievement",
"of": "of", "of": "of",
"sort": "Show not achieved first", "sort": "Show not achieved first",
"search": "Search achievement" "search": "Search achievement",
"commissions": "commissions",
"version": "Version",
"type": "Type",
"searchError": "Cannot found achievement: {query}"
}, },
"furnishing": { "furnishing": {
"title": "Furnishing", "title": "Furnishing",

View file

@ -665,7 +665,8 @@
"title": "Achievement", "title": "Achievement",
"of": "dari", "of": "dari",
"sort": "Urutkan yang belum didapatkan dulu", "sort": "Urutkan yang belum didapatkan dulu",
"search": "Cari achievement" "search": "Cari achievement",
"commissions": "misi harian"
}, },
"furnishing": { "furnishing": {
"title": "Furnitur", "title": "Furnitur",

View file

@ -9,15 +9,16 @@
import { locale, t } from 'svelte-i18n'; import { locale, t } from 'svelte-i18n';
import { onMount, tick } from 'svelte'; import { onMount, tick } from 'svelte';
import debounce from 'lodash/debounce'; import debounce from 'lodash/debounce';
import { mdiFilter } from '@mdi/js';
import Check from '../../components/Check.svelte'; import Check from '../../components/Check.svelte';
import Checkbox from '../../components/Checkbox.svelte'; import Checkbox from '../../components/Checkbox.svelte';
import { getAccountPrefix } from '../../stores/account'; import { getAccountPrefix } from '../../stores/account';
import { readSave, updateSave } from '../../stores/saveManager'; import { readSave, updateSave } from '../../stores/saveManager';
import Button from '../../components/Button.svelte'; import Button from '../../components/Button.svelte';
import Input from '../../components/Input.svelte';
import Icon from '../../components/Icon.svelte'; import Icon from '../../components/Icon.svelte';
import { mdiFilter } from '@mdi/js'; import Select from '../../components/Select.svelte';
import { pushToast } from '../../stores/toast';
export let data; export let data;
@ -41,6 +42,12 @@
let nameFilter = ''; let nameFilter = '';
let sortedAchievements = Object.entries(data).sort((a, b) => a[1].order - b[1].order); let sortedAchievements = Object.entries(data).sort((a, b) => a[1].order - b[1].order);
const versions = ['1.0', '1.1', '1.2', '1.3', '1.4', '1.5', '1.6', '2.0', '2.1'].map((e) => ({ label: e, value: e }));
let versionFilter = [];
const types = [{ label: $t('achievement.commissions'), value: 'commissions' }];
let typeFilter = [];
function parseCategories() { function parseCategories() {
categories = Object.entries(achievement) categories = Object.entries(achievement)
.map(([id, data]) => ({ .map(([id, data]) => ({
@ -116,53 +123,78 @@
return; return;
} }
const filterName = nameFilter !== '';
const query = nameFilter.toLowerCase(); const query = nameFilter.toLowerCase();
let index = 0; let index = 0;
let found = null;
for (const [id, item] of sortedAchievements) { for (const [id, item] of sortedAchievements) {
for (const achievement of item.achievements) { for (const achievement of item.achievements) {
if (Array.isArray(achievement)) { if (Array.isArray(achievement)) {
for (const e of achievement) { if (filterName && !achievement[0].name.toLowerCase().includes(query)) continue;
if (e.name.toLowerCase().includes(query)) { changeCategory(id, index, false);
found = achievement; return;
changeCategory(id, index, true, found); } else {
return; if (filterName && !achievement.name.toLowerCase().includes(query)) continue;
} changeCategory(id, index, false);
}
} else if (achievement.name.toLowerCase().includes(query)) {
found = achievement;
changeCategory(id, index, true, found);
return; return;
} }
} }
index++; index++;
} }
changeCategory(0, 0, true);
pushToast($t('achievement.searchError', { values: { query } }), 'error');
}, 500); }, 500);
async function changeCategory(id, index, firstLoad, search) { const updateSelectFilter = debounce(() => {
changeCategory(active, activeIndex, true);
}, 500);
async function changeCategory(id, index, firstLoad) {
active = id; active = id;
activeIndex = index; activeIndex = index;
const filterVersion = versionFilter.length > 0;
const filteredVersion = versionFilter.map((e) => e.value);
const filterComission = typeFilter.map((e) => e.value).includes('commissions');
const filterName = nameFilter !== '';
const query = nameFilter.toLowerCase();
console.log('filter', filterVersion, filterComission, filterName);
if (checkList[active] === undefined) { if (checkList[active] === undefined) {
checkList[active] = {}; checkList[active] = {};
} }
list = achievement[active].achievements.map((e) => { list = achievement[active].achievements
if (Array.isArray(e)) { .filter((e) => {
for (let i = 0; i < e.length; i++) { if (Array.isArray(e)) {
e[i].checked = checkList[active][e[i].id] === true; if (filterVersion && !filteredVersion.includes(e[0].ver)) return false;
if (filterComission && !e[0].commissions) return false;
if (filterName && !e[0].name.toLowerCase().includes(query)) return false;
return true;
} else {
if (filterVersion && !filteredVersion.includes(e.ver)) return false;
if (filterComission && !e.commissions) return false;
if (filterName && !e.name.toLowerCase().includes(query)) return false;
return true;
} }
return e; })
} else { .map((e) => {
e.checked = checkList[active][e.id] === true; if (Array.isArray(e)) {
return e; for (let i = 0; i < e.length; i++) {
} e[i].checked = checkList[active][e[i].id] === true;
}); }
return e;
} else {
e.checked = checkList[active][e.id] === true;
return e;
}
});
if (search) { if (sort) {
list = [search];
} else if (sort) {
originalList = list.slice(); originalList = list.slice();
list = list.sort((a, b) => { list = list.sort((a, b) => {
let first = a; let first = a;
@ -279,20 +311,22 @@
<img src="/images/primogem.png" class="w-4 h-4 ml-1" alt="primogem" /> <img src="/images/primogem.png" class="w-4 h-4 ml-1" alt="primogem" />
</div> </div>
</div> </div>
<Button <div class="flex space-x-2 items-center">
size="sm" <Button
on:click={() => { size="sm"
showFilter = !showFilter; on:click={() => {
}} showFilter = !showFilter;
> }}
<Icon path={mdiFilter} color="white" /> >
</Button> <Icon path={mdiFilter} color="white" />
<div class="lg:pl-4 text-white"> </Button>
<Checkbox checked={sort} on:change={() => changeSort(!sort)}>{$t('achievement.sort')}</Checkbox> <div class="pl-4 text-white">
<Checkbox checked={sort} on:change={() => changeSort(!sort)}>{$t('achievement.sort')}</Checkbox>
</div>
</div> </div>
</div> </div>
{#if showFilter} {#if showFilter}
<div> <div class="mb-2 flex flex-col md:flex-row space-y-2 md:space-y-0 md:space-x-2">
<div <div
class="flex flex-1 relative items-center bg-background rounded-2xl h-14 class="flex flex-1 relative items-center bg-background rounded-2xl h-14
focus-within:border-primary border-2 border-transparent ease-in duration-100" focus-within:border-primary border-2 border-transparent ease-in duration-100"
@ -305,6 +339,22 @@
class="pl-4 w-full min-h-full pr-4 text-white placeholder-gray-500 leading-none bg-transparent border-none focus:outline-none" class="pl-4 w-full min-h-full pr-4 text-white placeholder-gray-500 leading-none bg-transparent border-none focus:outline-none"
/> />
</div> </div>
<Select
multiselect
options={versions}
bind:selected={versionFilter}
placeholder={$t('achievement.version')}
className="w-full md:w-40"
on:change={updateSelectFilter}
/>
<Select
multiselect
options={types}
bind:selected={typeFilter}
placeholder={$t('achievement.type')}
className="w-full md:w-56"
on:change={updateSelectFilter}
/>
</div> </div>
{/if} {/if}
<div class="flex flex-col lg:flex-row space-y-3 lg:space-y-0 lg:space-x-3"> <div class="flex flex-col lg:flex-row space-y-3 lg:space-y-0 lg:space-x-3">
@ -338,7 +388,17 @@
{i > 0 && el[i - 1].checked !== true ? 'opacity-25' : ''}" {i > 0 && el[i - 1].checked !== true ? 'opacity-25' : ''}"
> >
<div class="flex-1 pr-1"> <div class="flex-1 pr-1">
<p class="font-semibold">{it.name}</p> <p class="font-semibold">
{it.name}
<span class="ml-1 rounded-xl bg-background px-2 text-gray-400 text-sm font-normal select-none">
{it.ver}
</span>
{#if it.commissions}
<span class="ml-1 rounded-xl bg-background px-2 text-gray-400 text-sm font-normal select-none">
{$t('achievement.commissions')}
</span>
{/if}
</p>
<p class="text-gray-400">{it.desc}</p> <p class="text-gray-400">{it.desc}</p>
</div> </div>
<div class="flex items-center"> <div class="flex items-center">
@ -359,7 +419,17 @@
{:else} {:else}
<div class="bg-item rounded-xl px-2 py-1 text-white flex items-center"> <div class="bg-item rounded-xl px-2 py-1 text-white flex items-center">
<div class="flex-1 pr-1"> <div class="flex-1 pr-1">
<p class="font-semibold">{el.name}</p> <p class="font-semibold">
{el.name}
<span class="ml-1 rounded-xl bg-background px-2 text-gray-400 text-sm font-normal select-none">
{el.ver}
</span>
{#if el.commissions}
<span class="ml-1 rounded-xl bg-background px-2 text-gray-400 text-sm font-normal select-none">
{$t('achievement.commissions')}
</span>
{/if}
</p>
<p class="text-gray-400">{el.desc}</p> <p class="text-gray-400">{el.desc}</p>
</div> </div>
<div class="flex items-center"> <div class="flex items-center">

View file

@ -252,7 +252,7 @@
<div class="lg:ml-64 pt-20 px-4 md:px-8 lg:pt-8"> <div class="lg:ml-64 pt-20 px-4 md:px-8 lg:pt-8">
<div class="bg-item rounded-xl mb-4 p-4"> <div class="bg-item rounded-xl mb-4 p-4">
<p class="text-white">{$t('settings.version')} <b>2.0</b></p> <p class="text-white">{$t('settings.version')} <b>2.1</b></p>
</div> </div>
<div class="bg-item rounded-xl mb-4 p-4 flex flex-col"> <div class="bg-item rounded-xl mb-4 p-4 flex flex-col">
<p class="text-white">{$t('settings.multiple')}</p> <p class="text-white">{$t('settings.multiple')}</p>