293 lines
5.3 KiB
Vue
293 lines
5.3 KiB
Vue
<template>
|
||
<div class="search-dialog">
|
||
<div class="dialog-cover" @click="closeDialog"></div>
|
||
<div class="dialog-content">
|
||
<button type="button" class="close-btn" @click="closeDialog">×</button>
|
||
<span class="title">搜索</span>
|
||
<input
|
||
type="text"
|
||
name=""
|
||
id="search-input"
|
||
placeholder="请输入关键字"
|
||
v-model="searchStr"
|
||
@input="search"
|
||
/>
|
||
<ul class="search-list">
|
||
<span>{{ status }}</span>
|
||
<li v-for="res in resultList" @click="closeDialog">
|
||
<a :href="base + res.href">{{ res.title }}</a>
|
||
</li>
|
||
</ul>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup lang="ts">
|
||
import { ref, onMounted } from 'vue'
|
||
|
||
const emit = defineEmits(['closeDialog'])
|
||
const closeDialog = (): void => {
|
||
// 添加关闭动画
|
||
const dialog = document.querySelector('.search-dialog') as HTMLElement
|
||
if (dialog) {
|
||
dialog.classList.add('hide-dialog')
|
||
setTimeout(() => {
|
||
emit('closeDialog')
|
||
}, 200)
|
||
}
|
||
}
|
||
|
||
import { data as posts } from '../../utils/posts.data'
|
||
import MiniSearch, { SearchResult } from 'minisearch'
|
||
import { useData } from 'vitepress'
|
||
|
||
const base = useData().site.value.base
|
||
const miniSearch = new MiniSearch({
|
||
fields: ['title', 'content'],
|
||
storeFields: ['title', 'href'],
|
||
searchOptions: {
|
||
fuzzy: 0.3,
|
||
},
|
||
})
|
||
miniSearch.addAll(posts)
|
||
|
||
const searchStr = defineModel<string>()
|
||
const resultList = ref<SearchResult[]>([])
|
||
const status = ref('这里空空的')
|
||
let timerId: ReturnType<typeof setTimeout> | null = null
|
||
|
||
function search(): void {
|
||
status.value = '搜索中……'
|
||
if (timerId) {
|
||
clearTimeout(timerId)
|
||
}
|
||
timerId = setTimeout(() => {
|
||
resultList.value = miniSearch.search(searchStr.value || '').slice(0, 5)
|
||
if (resultList.value.length) {
|
||
status.value = '搜到了~'
|
||
} else {
|
||
status.value = '这里空空的'
|
||
}
|
||
}, 500)
|
||
}
|
||
|
||
onMounted(() => {
|
||
const dialog = document.querySelector('.search-dialog') as HTMLElement
|
||
if (dialog) {
|
||
dialog.classList.add('show-dialog')
|
||
}
|
||
})
|
||
</script>
|
||
|
||
<style scoped lang="less">
|
||
.search-dialog {
|
||
position: fixed;
|
||
top: 0;
|
||
left: 0;
|
||
width: 100%;
|
||
height: 100%;
|
||
z-index: 200;
|
||
display: flex;
|
||
justify-content: center;
|
||
align-items: center;
|
||
opacity: 0;
|
||
animation: fadein 0.2s forwards;
|
||
}
|
||
// 遮罩
|
||
.dialog-cover {
|
||
background: rgba(0, 0, 0, 0.614);
|
||
position: absolute;
|
||
top: 0;
|
||
left: 0;
|
||
width: 100%;
|
||
height: 100%;
|
||
opacity: 0;
|
||
animation: cover-fadein 0.2s forwards;
|
||
}
|
||
// 搜索框
|
||
.dialog-content {
|
||
position: relative;
|
||
width: 90%;
|
||
max-width: 768px;
|
||
height: auto;
|
||
background-color: var(--search-dialog-bg);
|
||
border-radius: 16px;
|
||
padding: 10px;
|
||
overflow: hidden;
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
transform: scale(0.9);
|
||
opacity: 0;
|
||
animation: pop-up 0.2s forwards;
|
||
}
|
||
|
||
.dialog-content::before {
|
||
content: '';
|
||
position: absolute;
|
||
top: 0;
|
||
left: 0;
|
||
right: 0;
|
||
height: 56px;
|
||
border-bottom: 3px solid var(--search-dialog-border);
|
||
background-color: var(--search-dialog-header-bg);
|
||
background-image: var(--deco2);
|
||
background-repeat: no-repeat;
|
||
background-position: left;
|
||
background-size: contain;
|
||
}
|
||
|
||
.title {
|
||
font-weight: bold;
|
||
font-size: 25px;
|
||
padding: 5px 0;
|
||
border-bottom: 5px solid var(--font-color-gold);
|
||
z-index: 100;
|
||
}
|
||
|
||
.close-btn {
|
||
position: absolute;
|
||
top: 0;
|
||
right: 0;
|
||
width: 56px;
|
||
height: 56px;
|
||
font-size: 36px;
|
||
border: none;
|
||
background: transparent;
|
||
cursor: pointer;
|
||
}
|
||
|
||
#search-input {
|
||
color: var(--font-color-grey);
|
||
width: 100%;
|
||
height: 48px;
|
||
margin: 10px;
|
||
padding: 0 16px;
|
||
background-color: var(--search-input-bg);
|
||
border: 3px solid var(--search-input-border);
|
||
border-radius: 5px;
|
||
box-sizing: border-box;
|
||
|
||
&:focus {
|
||
outline: 3px solid var(--search-input-border);
|
||
}
|
||
}
|
||
|
||
.search-list {
|
||
width: 100%;
|
||
min-height: 100px;
|
||
box-sizing: border-box;
|
||
background-color: var(--search-list-bg);
|
||
border-radius: 8px;
|
||
padding: 10px;
|
||
margin: 0;
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 10px;
|
||
|
||
span {
|
||
text-align: center;
|
||
}
|
||
}
|
||
|
||
li {
|
||
background-color: var(--search-item-bg);
|
||
border-radius: 5px;
|
||
box-shadow: 3px 3px 3px var(--search-item-shadow);
|
||
padding: 10px;
|
||
height: 50px;
|
||
display: flex;
|
||
flex-direction: column;
|
||
justify-content: center;
|
||
|
||
a {
|
||
height: 50px;
|
||
line-height: 50px;
|
||
color: var(--font-color-grey);
|
||
}
|
||
}
|
||
|
||
@media (max-width: 768px) {
|
||
.dialog-content {
|
||
top: 5%;
|
||
}
|
||
}
|
||
|
||
// 动画
|
||
@keyframes fadein {
|
||
to {
|
||
opacity: 1;
|
||
}
|
||
}
|
||
|
||
@keyframes cover-fadein {
|
||
to {
|
||
opacity: 1;
|
||
}
|
||
}
|
||
|
||
@keyframes pop-up {
|
||
from {
|
||
opacity: 0;
|
||
transform: scale(0.9);
|
||
}
|
||
to {
|
||
opacity: 1;
|
||
transform: scale(1);
|
||
}
|
||
}
|
||
|
||
@keyframes fadeout {
|
||
from {
|
||
opacity: 1;
|
||
}
|
||
to {
|
||
opacity: 0;
|
||
}
|
||
}
|
||
|
||
@keyframes cover-fadeout {
|
||
from {
|
||
opacity: 1;
|
||
}
|
||
to {
|
||
opacity: 0;
|
||
}
|
||
}
|
||
|
||
.show-dialog {
|
||
animation: fadein 0.3s forwards;
|
||
}
|
||
|
||
.hide-dialog {
|
||
animation: fadeout 0.3s forwards;
|
||
}
|
||
|
||
.dialog-cover.show-dialog {
|
||
animation: cover-fadein 0.3s forwards;
|
||
}
|
||
|
||
.dialog-cover.hide-dialog {
|
||
animation: cover-fadeout 0.3s forwards;
|
||
}
|
||
|
||
.dialog-content.show-dialog {
|
||
animation: pop-up 0.3s forwards;
|
||
}
|
||
|
||
.dialog-content.hide-dialog {
|
||
animation: pop-down 0.3s forwards;
|
||
}
|
||
|
||
@keyframes pop-down {
|
||
from {
|
||
opacity: 1;
|
||
transform: scale(1);
|
||
}
|
||
to {
|
||
opacity: 0;
|
||
transform: scale(0.9);
|
||
}
|
||
}
|
||
</style>
|