Files
Aether/frontend/src/components/PlatformSelect.vue
fawney19 e9a6233655 refactor(frontend): 优化公共组件和布局组件
- 更新 Logo 相关组件 (AetherLogo, HeaderLogo, RippleLogo 等)
- 优化图表组件 (BarChart, LineChart, ScatterChart)
- 改进公共组件 (AlertDialog, EmptyState, LoadingState)
- 调整布局组件 (AppShell, SidebarNav, PageHeader 等)
- 优化 ActivityHeatmap 统计组件
2025-12-12 16:15:16 +08:00

331 lines
8.1 KiB
Vue

<template>
<div
ref="rootEl"
class="platform-select"
:class="[`platform-select--${sizeClass}` , { 'platform-select--open': isOpen }]"
tabindex="0"
@click="handleRootClick"
@keydown.enter.prevent="toggleDropdown"
@keydown.space.prevent="toggleDropdown"
@keydown.escape.stop="closeDropdown"
>
<div class="platform-select__current">
<component
:is="currentOption.icon"
class="platform-select__icon"
/>
<div class="platform-select__text">
<p class="platform-select__label">
{{ currentOption.label }}
</p>
<p class="platform-select__hint">
{{ currentOption.hint }}
</p>
</div>
</div>
<ChevronDown class="platform-select__chevron" />
<transition name="platform-select-fade">
<ul
v-if="isOpen"
class="platform-select__dropdown"
>
<li
v-for="option in resolvedOptions"
:key="option.value"
class="platform-select__option"
:class="{ 'platform-select__option--active': option.value === modelValue }"
@click.stop="selectOption(option.value)"
>
<component
:is="option.icon"
class="platform-select__option-icon"
/>
<div class="platform-select__option-copy">
<p class="platform-select__option-label">
{{ option.label }}
</p>
<p class="platform-select__option-hint">
{{ option.hint }}
</p>
</div>
<Check
v-if="option.value === modelValue"
class="platform-select__option-check"
/>
</li>
</ul>
</transition>
</div>
</template>
<script lang="ts">
import { Apple, Box, Monitor, Terminal } from 'lucide-vue-next'
import type { Component } from 'vue'
export interface PlatformOption {
value: string
label: string
hint: string
icon: Component
command: string
}
// Default options for backward compatibility
export const defaultPlatformOptions: PlatformOption[] = [
{ value: 'mac', label: 'Mac / Linux', hint: 'Terminal', icon: Terminal, command: '' },
{ value: 'windows', label: 'Windows', hint: 'PowerShell', icon: Monitor, command: '' }
]
// Preset configuration for each tool
export const platformPresets = {
default: {
options: defaultPlatformOptions,
defaultValue: 'mac'
},
claude: {
options: [
{ value: 'mac', label: 'Mac / Linux', hint: 'Terminal', icon: Terminal, command: 'curl -fsSL https://claude.ai/install.sh | bash' },
{ value: 'windows', label: 'Windows', hint: 'PowerShell', icon: Monitor, command: 'irm https://claude.ai/install.ps1 | iex' },
{ value: 'nodejs', label: 'Node.js', hint: 'npm', icon: Box, command: 'npm install -g @anthropic-ai/claude-code' },
{ value: 'homebrew', label: 'Mac', hint: 'Homebrew', icon: Apple, command: 'brew install --cask claude-code' }
] as PlatformOption[],
defaultValue: 'mac'
},
codex: {
options: [
{ value: 'nodejs', label: 'Node.js', hint: 'npm', icon: Box, command: 'npm install -g @openai/codex' },
{ value: 'homebrew', label: 'Mac', hint: 'Homebrew', icon: Apple, command: 'brew install --cask codex' }
] as PlatformOption[],
defaultValue: 'nodejs'
},
gemini: {
options: [
{ value: 'nodejs', label: 'Node.js', hint: 'npm', icon: Box, command: 'npm install -g @google/gemini-cli' },
{ value: 'homebrew', label: 'Mac', hint: 'Homebrew', icon: Apple, command: 'brew install gemini-cli' }
] as PlatformOption[],
defaultValue: 'nodejs'
}
}
// Helper to get command by platform value
export function getCommand(preset: keyof typeof platformPresets, value: string): string {
const config = platformPresets[preset]
return config.options.find((opt: PlatformOption) => opt.value === value)?.command ?? ''
}
</script>
<script setup lang="ts">
import { computed, onBeforeUnmount, onMounted, ref } from 'vue'
import { Check, ChevronDown } from 'lucide-vue-next'
const props = defineProps<{
modelValue: string
size?: 'md' | 'lg'
options?: PlatformOption[]
}>()
const emit = defineEmits<{
(event: 'update:modelValue', value: string): void
}>()
const rootEl = ref<HTMLElement | null>(null)
const isOpen = ref(false)
const sizeClass = computed(() => props.size ?? 'md')
const resolvedOptions = computed(() => props.options ?? defaultPlatformOptions)
const currentOption = computed(() => resolvedOptions.value.find((option: PlatformOption) => option.value === props.modelValue) ?? resolvedOptions.value[0])
function toggleDropdown() {
isOpen.value = !isOpen.value
}
function closeDropdown() {
isOpen.value = false
}
function selectOption(value: string) {
if (value !== props.modelValue) {
emit('update:modelValue', value)
}
closeDropdown()
}
function handleRootClick(event: MouseEvent) {
const dropdown = rootEl.value?.querySelector('.platform-select__dropdown')
if (dropdown?.contains(event.target as Node)) {
return
}
toggleDropdown()
}
function handleClickOutside(event: MouseEvent) {
if (!rootEl.value) {
return
}
if (!rootEl.value.contains(event.target as Node)) {
closeDropdown()
}
}
onMounted(() => {
document.addEventListener('click', handleClickOutside)
})
onBeforeUnmount(() => {
document.removeEventListener('click', handleClickOutside)
})
</script>
<style scoped>
.platform-select {
position: relative;
width: 11rem;
border: 1px solid var(--color-border);
border-radius: 0.9rem;
background-color: var(--color-background);
padding: 0.55rem 0.85rem;
cursor: pointer;
transition: border-color 0.2s ease, box-shadow 0.2s ease, background-color 0.2s ease;
display: flex;
align-items: center;
justify-content: space-between;
gap: 0.75rem;
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.12);
}
.dark .platform-select {
background-color: var(--color-background);
}
.platform-select--lg {
width: 13rem;
}
.platform-select:focus-visible,
.platform-select--open {
border-color: var(--color-primary);
box-shadow: 0 0 0 3px rgba(204, 120, 92, 0.2);
}
.platform-select__current {
display: flex;
align-items: center;
gap: 0.65rem;
}
.platform-select__icon {
width: 1.1rem;
height: 1.1rem;
color: var(--color-primary);
}
.platform-select__text {
display: flex;
flex-direction: column;
line-height: 1.1;
}
.platform-select__label {
font-size: 0.85rem;
font-weight: 600;
color: var(--color-text);
white-space: nowrap;
}
.platform-select__hint {
font-size: 0.7rem;
color: #91918d;
white-space: nowrap;
}
.dark .platform-select__hint {
color: #a8a29e;
}
.platform-select__chevron {
width: 0.9rem;
height: 0.9rem;
color: var(--color-border-soft);
}
.platform-select__dropdown {
position: absolute;
top: calc(100% + 0.45rem);
left: 0;
right: 0;
padding: 0.35rem;
border-radius: 1rem;
border: 1px solid var(--color-border);
background-color: var(--color-background);
box-shadow: 0 25px 55px rgba(0, 0, 0, 0.25);
z-index: 30;
backdrop-filter: blur(16px);
}
.platform-select__option {
display: flex;
align-items: center;
gap: 0.65rem;
padding: 0.55rem 0.6rem;
border-radius: 0.75rem;
transition: background 0.2s ease, color 0.2s ease;
}
.platform-select__option:hover {
background: rgba(204, 120, 92, 0.1);
}
.platform-select__option--active {
background: rgba(204, 120, 92, 0.18);
}
.platform-select__option-icon {
width: 1rem;
height: 1rem;
color: var(--color-primary);
}
.platform-select__option-copy {
display: flex;
flex-direction: column;
line-height: 1.1;
}
.platform-select__option-label {
font-size: 0.85rem;
font-weight: 600;
color: var(--color-text);
white-space: nowrap;
}
.platform-select__option-hint {
font-size: 0.7rem;
color: #91918d;
white-space: nowrap;
}
.dark .platform-select__option-hint {
color: #a8a29e;
}
.platform-select__option-check {
margin-left: auto;
width: 0.85rem;
height: 0.85rem;
color: var(--color-primary);
}
.platform-select-fade-enter-active,
.platform-select-fade-leave-active {
transition: opacity 0.15s ease, transform 0.15s ease;
}
.platform-select-fade-enter-from,
.platform-select-fade-leave-to {
opacity: 0;
transform: translateY(-6px);
}
</style>