完成部分设置项和游戏的代码

This commit is contained in:
expdsn 2024-10-25 18:53:08 +08:00
parent 199ca92fb3
commit 4f571eed13
16 changed files with 478 additions and 125 deletions

BIN
public/searchIcons/360.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.2 KiB

BIN
public/searchIcons/bing.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

View File

@ -10,14 +10,20 @@ import {
import AdderPageBack from './AdderPageBack' import AdderPageBack from './AdderPageBack'
import useLayoutStore from '../useLayoutStore' import useLayoutStore from '../useLayoutStore'
import { OhVueIcon, addIcons } from 'oh-vue-icons' import { OhVueIcon, addIcons } from 'oh-vue-icons'
import { MdKeyboardcommandkey, FaCompass, FaPencilRuler } from 'oh-vue-icons/icons' import {
MdKeyboardcommandkey,
FaCompass,
FaPencilRuler,
IoGameController
} from 'oh-vue-icons/icons'
import CustomAdder from './CustomAdder' import CustomAdder from './CustomAdder'
import clsx from 'clsx' import clsx from 'clsx'
import ThemeProvider from '@/utils/ThemeProvider' import ThemeProvider from '@/utils/ThemeProvider'
import WidgetAdder from './WidgetAdder' import WidgetAdder from './WidgetAdder'
import { Form, Input, Select } from 'ant-design-vue' import { Form, Input, Select } from 'ant-design-vue'
import HotAdder from './HotAdder' import HotAdder from './HotAdder'
addIcons(MdKeyboardcommandkey, FaCompass, FaPencilRuler) import GameAdder from './GameAdder'
addIcons(MdKeyboardcommandkey, FaCompass, FaPencilRuler, IoGameController)
const ItemButton = defineComponent({ const ItemButton = defineComponent({
props: { props: {
@ -118,6 +124,14 @@ export default defineComponent(() => {
type.value = 2 type.value = 2
}} }}
/> />
<ItemButton
name="io-game-controller"
label="快游戏"
active={type.value === 3}
onClick={() => {
type.value = 3
}}
/>
</div> </div>
<div <div
class={'w-0 h-full flex-grow relative z-10 flex flex-col ' + (isGame.value ? '' : '')} class={'w-0 h-full flex-grow relative z-10 flex flex-col ' + (isGame.value ? '' : '')}
@ -145,8 +159,10 @@ export default defineComponent(() => {
<WidgetAdder /> <WidgetAdder />
) : type.value === 1 ? ( ) : type.value === 1 ? (
<HotAdder /> <HotAdder />
) : ( ) : type.value === 2 ? (
<CustomAdder /> <CustomAdder />
) : (
<GameAdder />
)} )}
</Transition> </Transition>
</div> </div>

View File

@ -0,0 +1,211 @@
import CategoryTab from '@/utils/CategoryTab'
import request from '@/utils/request'
import { computed, defineComponent, inject, onMounted, ref, watch } from 'vue'
import type { BackgroundType } from '../background/BackgroundSwtich'
import clsx from 'clsx'
import useLayoutStore from '../useLayoutStore'
import { AddToToken } from './AdderPage'
import { v4 as uuid } from 'uuid'
import { Button } from 'ant-design-vue'
const URL_ADDRESS = 'http://newfatfox.oss-cn-beijing.aliyuncs.com'
interface GameType {
id: string
name: string
rom: string
playstation: string
hot: number
category: string
despt: string
icon: string
}
export const GameItem = defineComponent({
props: {
content: {
type: Object as () => GameType,
required: true
}
},
setup(props) {
const layout = useLayoutStore()
const isGame = computed(() => layout.state.current === 0)
const addTo = inject(AddToToken)
return () => (
<div
class={clsx(' w-full h-full rounded-lg flex flex-col justify-between shadow p-4', {
'bg-white/20': isGame.value,
'bg-white/80': !isGame.value
})}
key={props.content.name}
>
<div class="flex">
<img
src={props.content.icon.replace('res/game', URL_ADDRESS)}
class="w-[58px] h-[58px] bg-cover rounded-lg shadow-lg"
style={{
background: props.content.icon
}}
/>
<div class="px-2 w-0 flex-grow">
<div
class={clsx('text text-sm', {
'text-white': isGame.value,
'text-black/80': !isGame.value
})}
>
{props.content.name}
</div>
<div
class={clsx('text-[12px] line-clamp-2 text-ellipsis', {
'text-white/80': isGame.value,
'text-black/60': !isGame.value
})}
>
{props.content.despt}
</div>
</div>
</div>
<div class="flex justify-end">
<button
class={clsx(
'rounded-2xl text-[12px] flex justify-center items-center w-[60px] h-[24px]',
{
'bg-[#eeeeee] text-[#333] ': isGame.value,
'bg-[#ffaa4e] text-white': !isGame.value
}
)}
onClick={() => {
layout.addBlock(
{
id: uuid(),
link: '',
name: props.content.name,
label: props.content.name,
icon: '',
text: '',
background: '',
color: '',
w: 1,
h: 1
},
addTo?.value
)
}}
>
</button>
</div>
</div>
)
}
})
export default defineComponent(() => {
const layout = useLayoutStore()
const loading = ref(false)
const isGame = computed(() => layout.state.current === 0)
const appList = ref<GameType[]>([])
const selectType = ref('fc')
watch(
selectType,
(val) => {
loading.value = true
request<GameType[]>('GET', `/api/games?type=${val}`)
.then((res) => {
appList.value = res
})
.finally(() => {
setTimeout(() => {
loading.value = false
}, 200)
})
},
{
immediate: true
}
)
return () => (
<div class={'w-full h-full flex flex-col gap-y-4'}>
<div class={'w-full '}>
<CategoryTab
list={[
{
id: 'fc',
type: '经典红白机',
attr: 0,
oridinal: 0
},
{
id: 'md',
type: '经典世嘉',
attr: 1,
oridinal: 1
},
{
id: 'gma',
type: '经典GBA',
attr: 3,
oridinal: 3
}
]}
selectType={selectType.value}
onUpdate:type={(e) => {
selectType.value = e
}}
v-slots={{
select: (text: string) => (
<button
class={clsx(
'px-[20px] py-1 items-center justify-center duration-75 shrink-0 flex rounded-2xl whitespace-nowrap ',
isGame.value
? 'bg-white/30 text-white bg-gradient-to-b from-[#ffaa4e] to-[#ff6227]'
: 'text-[#000] bg-[#D6D6D6]'
)}
>
{text}
</button>
),
unSelect: (text: string) => (
<button
class={clsx(
'px-[20px] py-1 items-center justify-center duration-75 shrink-0 flex rounded-2xl whitespace-nowrap',
isGame.value
? ' text-[#999] bg-white/10 hover:bg-white/20'
: 'text-[#000] hover:bg-[#f0ecec]'
)}
>
{text}
</button>
)
}}
></CategoryTab>
</div>
<div class={'h-0 flex-1 w-full'}>
<div class={'w-full h-full overflow-y-scroll scrollbar-hide'}>
{!loading.value ? (
<div class={'w-full grid grid-cols-3 gap-4 '} style="grid-auto-rows: 120px">
{appList.value.map((el) => (
<GameItem content={el}></GameItem>
))}
</div>
) : (
<div class={'w-full grid grid-cols-3 gap-4 '} style="grid-auto-rows: 120px">
{Array(12)
.fill(0)
.map((el, idx) => (
<div
class={
' relative cursor-pointer bg-gray-500 group w-full flex-grow-0 rounded-xl overflow-hidden'
}
key={idx}
/>
))}
</div>
)}
</div>
</div>
</div>
)
})

View File

@ -5,6 +5,7 @@ import type { BackgroundType } from '../background/BackgroundSwtich'
import clsx from 'clsx' import clsx from 'clsx'
import useLayoutStore from '../useLayoutStore' import useLayoutStore from '../useLayoutStore'
import { AddToToken } from './AdderPage' import { AddToToken } from './AdderPage'
import { v4 as uuid } from 'uuid'
import { Button } from 'ant-design-vue' import { Button } from 'ant-design-vue'
interface HotAppCategoryType { interface HotAppCategoryType {
@ -34,14 +35,21 @@ export const LinkItem = defineComponent({
const addTo = inject(AddToToken) const addTo = inject(AddToToken)
return () => ( return () => (
<div <div
class={clsx('bg-white w-full h-full rounded-lg shadow p-4', { class={clsx(' w-full h-full rounded-lg flex flex-col justify-between shadow p-4', {
'bg-white/20': isGame.value, 'bg-white/20': isGame.value,
'bg-white/80': !isGame.value 'bg-white/80': !isGame.value
})} })}
key={props.content.name} key={props.content.name}
> >
<div class="flex"> <div class="flex">
<img src={props.content.icon} class="w-[48px] h-[48px] bg-cover rounded-lg" /> <img
src={props.content.icon}
class="w-[58px] h-[58px] bg-cover rounded-lg shadow-lg"
style={{
background: props.content.background,
}}
/>
<div class="px-2 w-0 flex-grow"> <div class="px-2 w-0 flex-grow">
<div <div
class={clsx('text text-sm', { class={clsx('text text-sm', {
@ -62,9 +70,34 @@ export const LinkItem = defineComponent({
</div> </div>
</div> </div>
<div class="flex justify-end"> <div class="flex justify-end">
<Button size="small" type="primary" onClick={() => {}}> <button
class={clsx(
'rounded-2xl text-[12px] flex justify-center items-center w-[60px] h-[24px]',
{
'bg-[#eeeeee] text-[#333] ': isGame.value,
'bg-[#ffaa4e] text-white': !isGame.value
}
)}
onClick={() => {
layout.addBlock(
{
id: uuid(),
link: '',
name: props.content.name,
label: props.content.name,
icon: '',
text: '',
background: '',
color: '',
w: 1,
h: 1
},
addTo?.value
)
}}
>
</Button> </button>
</div> </div>
</div> </div>
) )
@ -130,7 +163,9 @@ export default defineComponent(() => {
<button <button
class={clsx( class={clsx(
'px-[20px] py-1 items-center justify-center duration-75 shrink-0 flex rounded-2xl whitespace-nowrap', 'px-[20px] py-1 items-center justify-center duration-75 shrink-0 flex rounded-2xl whitespace-nowrap',
isGame.value ? ' text-[#999] bg-white/10 hover:bg-white/20' : 'text-[#000] hover:bg-[#f0ecec]' isGame.value
? ' text-[#999] bg-white/10 hover:bg-white/20'
: 'text-[#000] hover:bg-[#f0ecec]'
)} )}
> >
{text} {text}
@ -142,19 +177,19 @@ export default defineComponent(() => {
<div class={'h-0 flex-1 w-full'}> <div class={'h-0 flex-1 w-full'}>
<div class={'w-full h-full overflow-y-scroll scrollbar-hide'}> <div class={'w-full h-full overflow-y-scroll scrollbar-hide'}>
{!loading.value ? ( {!loading.value ? (
<div class={'w-full grid grid-cols-3 gap-4 '} style="grid-auto-rows: 100px"> <div class={'w-full grid grid-cols-3 gap-4 '} style="grid-auto-rows: 120px">
{appList.value.map((el) => ( {appList.value.map((el) => (
<LinkItem content={el}></LinkItem> <LinkItem content={el}></LinkItem>
))} ))}
</div> </div>
) : ( ) : (
<div class={'w-full grid grid-cols-3 gap-4 '} style="grid-auto-rows: 100px"> <div class={'w-full grid grid-cols-3 gap-4 '} style="grid-auto-rows: 120px">
{Array(12) {Array(12)
.fill(0) .fill(0)
.map((el, idx) => ( .map((el, idx) => (
<div <div
class={ class={
'h-[100px] relative cursor-pointer bg-gray-500 group w-full flex-grow-0 rounded-xl overflow-hidden' ' relative cursor-pointer bg-gray-500 group w-full flex-grow-0 rounded-xl overflow-hidden'
} }
key={idx} key={idx}
/> />

View File

@ -26,26 +26,30 @@ export default defineComponent(() => {
label: () => <div></div> label: () => <div></div>
}} }}
> >
<div class={'w-[180px]'}>
<Slider <Slider
v-model:value={settings.state.blockRadius} v-model:value={settings.state.blockRadius}
step={0.1} step={0.01}
tooltipOpen={false} tooltipOpen={false}
min={0} min={0}
max={1} max={1}
/> />
</div>
</SettingItem>{' '} </SettingItem>{' '}
<SettingItem <SettingItem
v-slots={{ v-slots={{
label: () => <div></div> label: () => <div></div>
}} }}
> >
<div class={'w-[180px]'}>
<Slider <Slider
v-model:value={settings.state.mainWidth} v-model:value={settings.state.mainWidth}
step={1} step={0.01}
tooltipOpen={false} tooltipOpen={false}
min={40} min={0}
max={80} max={1}
/> />
</div>
</SettingItem> </SettingItem>
{/* <SettingItem {/* <SettingItem
noRoundedB noRoundedB

View File

@ -0,0 +1,61 @@
import SettingItem from '@/settings/SettingItem'
import useSettingsStore from '@/settings/useSettingsStore'
import { Select, Slider, Switch, type SelectProps } from 'ant-design-vue'
import { defineComponent, ref } from 'vue'
import useSearchConfigStore from '../header/search/useSearchConfigStore'
export default defineComponent({
name: 'SearchSetting',
props: {},
setup() {
const settings = useSettingsStore()
const searchStore = useSearchConfigStore()
const options1 = ref<SelectProps['options']>(
[...searchStore.defaultList, ...searchStore.customList].map((item) => ({
value: item.name,
label: item.name
}))
)
return () => (
<div class="p-4">
<SettingItem
noRoundedB
v-slots={{
label: () => <div></div>
}}
>
<Switch v-model:checked={settings.state.showBlockLabel} />
</SettingItem>
<SettingItem
noRoundedT
v-slots={{
label: () => <div></div>
}}
>
<Switch v-model:checked={settings.state.showHistory} />
</SettingItem>
<SettingItem
v-slots={{
label: () => <div></div>
}}
>
<Switch v-model:checked={settings.state.showHistory} />
</SettingItem>{' '}
<SettingItem
v-slots={{
label: () => <div></div>
}}
>
<Select
value={searchStore.current.name}
onUpdate:value={(e) => {
searchStore.selectSearchByName(e as string)
}}
style={{ width: 100 }}
options={options1.value}
/>
</SettingItem>
</div>
)
}
})

View File

@ -3,11 +3,14 @@ import { OhVueIcon, addIcons } from 'oh-vue-icons'
import { MdHistory, MdRemove } from 'oh-vue-icons/icons' import { MdHistory, MdRemove } from 'oh-vue-icons/icons'
import useSearchConfigStore from './useSearchConfigStore' import useSearchConfigStore from './useSearchConfigStore'
import jump from '@/utils/jump' import jump from '@/utils/jump'
import useSettingsStore from '@/settings/useSettingsStore'
addIcons(MdHistory) addIcons(MdHistory)
addIcons(MdRemove) addIcons(MdRemove)
export default defineComponent(() => { export default defineComponent(() => {
const searchConfig = useSearchConfigStore() const searchConfig = useSearchConfigStore()
return () => ( const settings = useSettingsStore()
return () =>
settings.state.showHistory && (
<div class="absolute left-0 -bottom-2 translate-y-full w-full rounded-lg bg-white/60 backdrop-blur shadow-lg p-4"> <div class="absolute left-0 -bottom-2 translate-y-full w-full rounded-lg bg-white/60 backdrop-blur shadow-lg p-4">
{searchConfig.history.map((item, idx) => ( {searchConfig.history.map((item, idx) => (
<div <div

View File

@ -22,7 +22,7 @@ export default defineComponent(() => {
> >
<div <div
class={ class={
'w-full h-11 shadow-content overflow-hidden max-w-[90vw] transition-all flex justify-between items-center gap-4 ' + 'w-full h-11 shadow-content overflow-hidden max-w-[90vw] px-1 transition-all flex justify-between items-center gap-4 ' +
(search.focus ? 'bg-white/60' : 'bg-white/40 hover:bg-white/60') (search.focus ? 'bg-white/60' : 'bg-white/40 hover:bg-white/60')
} }
style={{ style={{
@ -30,18 +30,19 @@ export default defineComponent(() => {
}} }}
> >
<div <div
class="w-11 h-11 flex justify-center items-center" class="w-9 h-11 flex justify-center items-center cursor-pointer"
onClick={() => { onClick={() => {
search.showSearchConfig = true search.showSearchConfig = true
}} }}
> >
<div class="p-2 rounded-full overflow-hidden bg-white/40 cursor-pointer">
<div <div
class="w-4 h-4 bg-center bg-contain bg-no-repeat" class="w-[26px] h-[26px] bg-center bg-contain bg-no-repeat"
style={{ backgroundImage: `url('${searchConfig.current.icon}')` }} style={{
backgroundImage: `url('${searchConfig.current.icon}')`,
borderRadius: settings.state.searchRadius - 4 + 'px'
}}
/> />
</div> </div>
</div>
<input <input
v-model={search.searchStr} v-model={search.searchStr}
ref={(el) => (search.searchRef = el as any)} ref={(el) => (search.searchRef = el as any)}

View File

@ -12,78 +12,79 @@ const defaultSearchList: SearchInfo[] = [
{ {
name: '百度', name: '百度',
url: 'https://www.baidu.com/s?wd=', url: 'https://www.baidu.com/s?wd=',
icon: 'searchIcons/baidu.svg', icon: 'searchIcons/baidu.png',
show: true show: true
}, },
{ {
name: '必应', name: '必应',
url: 'https://cn.bing.com/search?q=', url: 'https://cn.bing.com/search?q=',
icon: 'searchIcons/bing.svg', icon: 'searchIcons/bing.png',
show: true show: true
}, },
{ {
name: '谷歌', name: '谷歌',
url: 'https://www.google.com/search?q=', url: 'https://www.google.com/search?q=',
icon: 'searchIcons/google.svg', icon: 'searchIcons/google.png',
show: true show: true
}, },
{ {
name: '360', name: '360',
url: 'https://www.so.com/s?q=', url: 'https://www.so.com/s?q=',
icon: 'searchIcons/360.svg', icon: 'searchIcons/360.png',
show: true
},
{
name: '搜狗',
url: 'https://www.sogou.com/sogou?&query=',
icon: 'searchIcons/sougou.svg',
show: true show: true
} }
] ]
const defaultCustomSearchList: SearchInfo[] = [ const defaultCustomSearchList: SearchInfo[] = [
{
name: '知乎',
url: 'https://www.zhihu.com/search?type=content&q=',
icon: 'searchIcons/zhihu.svg',
show: true
},
{
name: 'GitHub',
url: 'https://github.com/search?q=',
icon: 'searchIcons/GitHub.svg',
show: true
},
{
name: 'F搜',
url: 'https://fsoufsou.com/search?q=',
icon: 'searchIcons/F.svg',
show: true
},
{
name: '豆瓣',
url: 'https://www.douban.com/search?q=',
icon: 'searchIcons/douban.svg',
show: true
},
{
name: 'Yandex',
url: 'https://yandex.com/search/?text=',
icon: 'searchIcons/yandex.svg',
show: true
},
{
name: '开发者',
url: 'https://kaifa.baidu.com/searchPage?wd=',
icon: 'searchIcons/kaifa.svg',
show: true
},
{
name: 'B站',
url: 'https://search.bilibili.com/all?keyword=',
icon: 'searchIcons/bilibili.svg',
show: true
}
] ]
// const defaultCustomSearchList: SearchInfo[] = [
// {
// name: '知乎',
// url: 'https://www.zhihu.com/search?type=content&q=',
// icon: 'searchIcons/zhihu.svg',
// show: true
// },
// {
// name: 'GitHub',
// url: 'https://github.com/search?q=',
// icon: 'searchIcons/GitHub.svg',
// show: true
// },
// {
// name: 'F搜',
// url: 'https://fsoufsou.com/search?q=',
// icon: 'searchIcons/F.svg',
// show: true
// },
// {
// name: '豆瓣',
// url: 'https://www.douban.com/search?q=',
// icon: 'searchIcons/douban.svg',
// show: true
// },
// {
// name: 'Yandex',
// url: 'https://yandex.com/search/?text=',
// icon: 'searchIcons/yandex.svg',
// show: true
// },
// {
// name: '开发者',
// url: 'https://kaifa.baidu.com/searchPage?wd=',
// icon: 'searchIcons/kaifa.svg',
// show: true
// },
// {
// name: 'B站',
// url: 'https://search.bilibili.com/all?keyword=',
// icon: 'searchIcons/bilibili.svg',
// show: true
// }
// ]
export default defineStore( export default defineStore(
'searchConfig', 'searchConfig',
() => { () => {
@ -100,13 +101,20 @@ export default defineStore(
const removeHistory = (idx: number) => { const removeHistory = (idx: number) => {
history.value.splice(idx, 1) history.value.splice(idx, 1)
} }
const selectSearchByName = (name: string) => {
const search = defaultList.value.find((item) => item.name === name)
if (search) {
current.value = search
}
}
return { return {
current, current,
customList, customList,
defaultList, defaultList,
history, history,
addHistory, addHistory,
removeHistory removeHistory,
selectSearchByName
} }
}, },
{ persist: true } { persist: true }

View File

@ -1,23 +1,26 @@
import useRouterStore from '@/useRouterStore' import useRouterStore from '@/useRouterStore'
import { defineComponent, Transition } from 'vue' import { defineComponent } from 'vue'
import UserPage from '@/user/UserPage' import UserPage from '@/user/UserPage'
import BackgroundPage from '@/layout/background/BackgroundPage' import BackgroundPage from '@/layout/background/BackgroundPage'
import ThemeProvider from '@/utils/ThemeProvider' import ThemeProvider from '@/utils/ThemeProvider'
import BlockSettings from '@/layout/grid/BlockSettings' import BlockSettings from '@/layout/grid/BlockSettings'
import SearchSetting from '@/layout/grid/SearchSetting'
export default defineComponent(() => { export default defineComponent(() => {
const router = useRouterStore() const router = useRouterStore()
return () => ( return () => (
<div class="w-0 h-full flex-grow bg-white/80 backdrop-blur"> <div class="w-0 h-full flex-grow bg-white/80 backdrop-blur">
<ThemeProvider> <ThemeProvider>
<Transition>
{router.path === 'settings-user' ? ( {router.path === 'settings-user' ? (
<UserPage /> <UserPage />
) : router.path === 'settings-background' ? ( ) : router.path === 'settings-background' ? (
<BackgroundPage /> <BackgroundPage />
) : router.path === 'settings-block' ? ( ) : router.path === 'settings-block' ? (
<BlockSettings /> <BlockSettings />
) : router.path === 'settings-search' ? (
<SearchSetting />
) : null} ) : null}
</Transition>
</ThemeProvider> </ThemeProvider>
</div> </div>
) )

View File

@ -18,6 +18,7 @@ export default defineStore(
showDate: true, showDate: true,
showTime: true, showTime: true,
showAdder: true, showAdder: true,
showHistory: true,
// 尺寸 // 尺寸
blockSize: 6.7, blockSize: 6.7,
blockPadding: 1, blockPadding: 1,
@ -26,7 +27,8 @@ export default defineStore(
showBlockLabel: true, showBlockLabel: true,
// 搜索 // 搜索
searchWidth: 30, searchWidth: 30,
searchRadius: 24 searchRadius: 12
}) })
return { state, blockInner: computed(() => state.blockSize - 2 * state.blockPadding) } return { state, blockInner: computed(() => state.blockSize - 2 * state.blockPadding) }
}, },

View File

@ -15,9 +15,18 @@ export default defineComponent({
theme={{ theme={{
algorithm: props.dark ? theme.darkAlgorithm : theme.defaultAlgorithm, algorithm: props.dark ? theme.darkAlgorithm : theme.defaultAlgorithm,
token: { token: {
colorPrimary: '#f7a94e', colorPrimary: '#f88e14',
colorBgBase: props.dark ? '#393a41' : '#fff', colorBgBase: props.dark ? '#393a41' : '#fff',
colorBorder: 'transparent' colorBorder: 'transparent'
},
components: {
Slider: {
},
Button: {
colorPrimary: '#f88e14'
}
} }
}} }}
locale={zhCN} locale={zhCN}