hippofish/packages/client/src/components/MkInstanceSelectDialog.vue

186 lines
3.4 KiB
Vue
Raw Normal View History

<template>
2023-04-08 02:01:42 +02:00
<XModalWindow
ref="dialogEl"
:with-ok-button="true"
:ok-button-disabled="selected == null"
@click="cancel()"
@close="cancel()"
@ok="ok()"
@closed="$emit('closed')"
>
<template #header>{{ i18n.ts.selectInstance }}</template>
<div class="mehkoush">
<div class="form">
<MkInput
v-model="hostname"
:autofocus="true"
@update:modelValue="search"
>
<template #label>{{ i18n.ts.instance }}</template>
</MkInput>
2023-04-08 02:01:42 +02:00
</div>
<div
v-if="hostname != ''"
class="result"
:class="{ hit: instances.length > 0 }"
>
<div v-if="instances.length > 0" class="instances">
<div
v-for="item in instances"
:key="item.id"
class="instance"
:class="{
selected: selected && selected.id === item.id,
}"
@click="selected = item"
@dblclick="ok()"
>
<div class="body">
<img
class="icon"
:src="
item.iconUrl ?? '/client-assets/dummy.png'
"
aria-hidden="true"
/>
<span class="name">{{ item.host }}</span>
</div>
</div>
</div>
2023-04-08 02:01:42 +02:00
<div v-else class="empty">
<span>{{ i18n.ts.noInstances }}</span>
</div>
</div>
</div>
2023-04-08 02:01:42 +02:00
</XModalWindow>
</template>
<script lang="ts" setup>
import { ref } from "vue";
import type { entities } from "firefish-js";
2023-04-08 02:01:42 +02:00
import MkInput from "@/components/form/input.vue";
import XModalWindow from "@/components/MkModalWindow.vue";
import * as os from "@/os";
import { i18n } from "@/i18n";
const emit = defineEmits<{
2024-04-11 19:12:18 +02:00
ok: [selected: entities.Instance];
cancel: [];
closed: [];
}>();
2023-09-02 01:27:33 +02:00
const hostname = ref("");
const instances = ref<entities.Instance[]>([]);
2024-04-11 19:12:18 +02:00
const selected = ref<entities.Instance | null>(null);
2023-09-02 01:27:33 +02:00
const dialogEl = ref<InstanceType<typeof XModalWindow>>();
let searchOrderLatch = 0;
const search = () => {
if (hostname.value === "") {
instances.value = [];
return;
}
const searchId = ++searchOrderLatch;
2023-04-08 02:01:42 +02:00
os.api("federation/instances", {
host: hostname.value,
limit: 10,
blocked: false,
suspended: false,
2023-04-08 02:01:42 +02:00
sort: "+pubSub",
}).then((_instances) => {
if (searchId !== searchOrderLatch) return;
instances.value = _instances.map(
2023-04-08 02:01:42 +02:00
(x) =>
({
id: x.id,
host: x.host,
iconUrl: x.iconUrl,
}) as entities.Instance,
2023-04-08 02:01:42 +02:00
);
});
};
const ok = () => {
if (selected.value == null) return;
emit("ok", selected.value);
dialogEl.value?.close();
};
const cancel = () => {
2023-04-08 02:01:42 +02:00
emit("cancel");
dialogEl.value?.close();
};
</script>
<style lang="scss" scoped>
.mehkoush {
margin: var(--marginFull) 0;
> .form {
padding: 0 var(--root-margin);
}
> .result {
display: flex;
flex-direction: column;
overflow: auto;
height: 100%;
&.result.hit {
padding: 0;
}
> .instances {
flex: 1;
overflow: auto;
padding: 8px 0;
> .instance {
display: flex;
align-items: center;
padding: 8px var(--root-margin);
font-size: 14px;
&:hover {
background: var(--X7);
}
&.selected {
background: var(--accent);
color: #fff;
}
> * {
pointer-events: none;
user-select: none;
}
> .body {
padding: 0 8px;
width: 100%;
> .name {
display: block;
font-weight: bold;
}
> .icon {
width: 16px;
height: 16px;
margin-right: 8px;
float: left;
}
}
}
}
> .empty {
opacity: 0.7;
text-align: center;
}
}
}
</style>