2023-11-30 14:49:26 +09:00
|
|
|
<!--
|
2024-02-14 00:59:27 +09:00
|
|
|
SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-11-30 14:49:26 +09:00
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
-->
|
|
|
|
|
|
|
|
<template>
|
|
|
|
<MkContainer :showHeader="widgetProps.showHeader" class="mkw-bdayfollowings">
|
|
|
|
<template #icon><i class="ti ti-cake"></i></template>
|
|
|
|
<template #header>{{ i18n.ts._widgets.birthdayFollowings }}</template>
|
2024-03-25 18:31:30 +09:00
|
|
|
<template #func="{ buttonStyleClass }"><button class="_button" :class="buttonStyleClass" @click="actualFetch()"><i class="ti ti-refresh"></i></button></template>
|
2023-11-30 14:49:26 +09:00
|
|
|
|
|
|
|
<div :class="$style.bdayFRoot">
|
|
|
|
<MkLoading v-if="fetching"/>
|
|
|
|
<div v-else-if="users.length > 0" :class="$style.bdayFGrid">
|
|
|
|
<MkAvatar v-for="user in users" :key="user.id" :user="user.followee" link preview></MkAvatar>
|
|
|
|
</div>
|
|
|
|
<div v-else :class="$style.bdayFFallback">
|
|
|
|
<img :src="infoImageUrl" class="_ghost" :class="$style.bdayFFallbackImage"/>
|
|
|
|
<div>{{ i18n.ts.nothing }}</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</MkContainer>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts" setup>
|
|
|
|
import { ref } from 'vue';
|
|
|
|
import * as Misskey from 'misskey-js';
|
|
|
|
import { useWidgetPropsManager, WidgetComponentEmits, WidgetComponentExpose, WidgetComponentProps } from './widget.js';
|
|
|
|
import { GetFormResultType } from '@/scripts/form.js';
|
|
|
|
import MkContainer from '@/components/MkContainer.vue';
|
2024-01-04 18:32:46 +09:00
|
|
|
import { misskeyApi } from '@/scripts/misskey-api.js';
|
2023-11-30 14:49:26 +09:00
|
|
|
import { useInterval } from '@/scripts/use-interval.js';
|
|
|
|
import { i18n } from '@/i18n.js';
|
|
|
|
import { infoImageUrl } from '@/instance.js';
|
|
|
|
import { $i } from '@/account.js';
|
|
|
|
|
|
|
|
const name = i18n.ts._widgets.birthdayFollowings;
|
|
|
|
|
|
|
|
const widgetPropsDef = {
|
|
|
|
showHeader: {
|
|
|
|
type: 'boolean' as const,
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
type WidgetProps = GetFormResultType<typeof widgetPropsDef>;
|
|
|
|
|
|
|
|
const props = defineProps<WidgetComponentProps<WidgetProps>>();
|
|
|
|
const emit = defineEmits<WidgetComponentEmits<WidgetProps>>();
|
|
|
|
|
|
|
|
const { widgetProps, configure } = useWidgetPropsManager(name,
|
|
|
|
widgetPropsDef,
|
|
|
|
props,
|
|
|
|
emit,
|
|
|
|
);
|
|
|
|
|
2024-03-25 18:31:30 +09:00
|
|
|
const users = ref<Misskey.Endpoints['users/following']['res']>([]);
|
2023-11-30 14:49:26 +09:00
|
|
|
const fetching = ref(true);
|
|
|
|
let lastFetchedAt = '1970-01-01';
|
|
|
|
|
|
|
|
const fetch = () => {
|
|
|
|
if (!$i) {
|
|
|
|
users.value = [];
|
|
|
|
fetching.value = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const lfAtD = new Date(lastFetchedAt);
|
|
|
|
lfAtD.setHours(0, 0, 0, 0);
|
|
|
|
const now = new Date();
|
|
|
|
now.setHours(0, 0, 0, 0);
|
|
|
|
|
|
|
|
if (now > lfAtD) {
|
2024-03-25 18:31:30 +09:00
|
|
|
actualFetch();
|
2023-11-30 14:49:26 +09:00
|
|
|
|
|
|
|
lastFetchedAt = now.toISOString();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-03-25 18:31:30 +09:00
|
|
|
function actualFetch() {
|
|
|
|
if ($i == null) {
|
|
|
|
users.value = [];
|
|
|
|
fetching.value = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const now = new Date();
|
|
|
|
now.setHours(0, 0, 0, 0);
|
|
|
|
fetching.value = true;
|
|
|
|
misskeyApi('users/following', {
|
|
|
|
limit: 18,
|
|
|
|
birthday: `${now.getFullYear().toString().padStart(4, '0')}-${(now.getMonth() + 1).toString().padStart(2, '0')}-${now.getDate().toString().padStart(2, '0')}`,
|
|
|
|
userId: $i.id,
|
|
|
|
}).then(res => {
|
|
|
|
users.value = res;
|
|
|
|
window.setTimeout(() => {
|
|
|
|
// 早すぎるとチカチカする
|
|
|
|
fetching.value = false;
|
|
|
|
}, 100);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-11-30 14:49:26 +09:00
|
|
|
useInterval(fetch, 1000 * 60, {
|
|
|
|
immediate: true,
|
|
|
|
afterMounted: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
defineExpose<WidgetComponentExpose>({
|
|
|
|
name,
|
|
|
|
configure,
|
|
|
|
id: props.widget ? props.widget.id : null,
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" module>
|
|
|
|
.bdayFRoot {
|
|
|
|
overflow: hidden;
|
|
|
|
min-height: calc(calc(calc(50px * 3) - 8px) + calc(var(--margin) * 2));
|
|
|
|
}
|
|
|
|
.bdayFGrid {
|
|
|
|
display: grid;
|
|
|
|
grid-template-columns: repeat(6, 42px);
|
|
|
|
grid-template-rows: repeat(3, 42px);
|
|
|
|
place-content: center;
|
|
|
|
gap: 8px;
|
|
|
|
margin: var(--margin) auto;
|
|
|
|
}
|
|
|
|
|
|
|
|
.bdayFFallback {
|
|
|
|
height: 100%;
|
|
|
|
display: flex;
|
|
|
|
flex-direction: column;
|
|
|
|
justify-content: center;
|
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
.bdayFFallbackImage {
|
|
|
|
height: 96px;
|
|
|
|
width: auto;
|
|
|
|
max-width: 90%;
|
|
|
|
margin-bottom: 8px;
|
|
|
|
border-radius: var(--radius);
|
|
|
|
}
|
|
|
|
</style>
|