2026-01-18 12:21:49 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="syntax-comparison-demo">
|
|
|
|
|
|
<div class="demo-header">
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<span class="icon">📝</span>
|
|
|
|
|
|
<span class="title">语法对比镜</span>
|
|
|
|
|
|
<span class="subtitle">同样的功能,不同的表达方式</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<div class="intro-text">
|
|
|
|
|
|
想象你在<span class="highlight">写信</span>:有人喜欢简洁明了(Python、Ruby),有人喜欢正式严谨(Java、C#),有人喜欢直接高效(Go)。不同语言的语法反映了不同的设计哲学。
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="language-tabs">
|
2026-01-18 12:21:49 +08:00
|
|
|
|
<button
|
|
|
|
|
|
v-for="lang in languages"
|
|
|
|
|
|
:key="lang.name"
|
2026-02-13 22:10:03 +08:00
|
|
|
|
class="lang-tab"
|
2026-01-18 12:21:49 +08:00
|
|
|
|
:class="{ active: selectedLang === lang.name }"
|
|
|
|
|
|
@click="selectedLang = lang.name"
|
|
|
|
|
|
>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<span class="tab-icon">{{ lang.icon }}</span>
|
|
|
|
|
|
<span class="tab-name">{{ lang.name }}</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-18 17:38:10 +08:00
|
|
|
|
<Transition
|
|
|
|
|
|
name="fade"
|
|
|
|
|
|
mode="out-in"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div
|
|
|
|
|
|
:key="selectedLang"
|
|
|
|
|
|
class="code-display"
|
|
|
|
|
|
>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<div class="code-window">
|
|
|
|
|
|
<div class="window-header">
|
|
|
|
|
|
<div class="window-controls">
|
2026-02-18 17:38:10 +08:00
|
|
|
|
<span class="control red" />
|
|
|
|
|
|
<span class="control yellow" />
|
|
|
|
|
|
<span class="control green" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="file-name">
|
|
|
|
|
|
{{ getCode(selectedLang).filename }}
|
2026-02-13 22:10:03 +08:00
|
|
|
|
</div>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<pre class="code-content">{{ getCode(selectedLang).code }}</pre>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="code-stats">
|
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
|
<span class="stat-label">代码行数:</span>
|
|
|
|
|
|
<span class="stat-value">{{ getLineCount(selectedLang) }} 行</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
|
<span class="stat-label">复杂度:</span>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<span class="stat-value">{{ getCode(selectedLang).complexity }}</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
</Transition>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<div class="info-box">
|
|
|
|
|
|
<span class="icon">💡</span>
|
|
|
|
|
|
<strong>核心思想:</strong>简洁的语法(Python、Ruby)让开发更快,但冗长的语法(Java、C#)提供了更强的类型安全性和可维护性。没有"最好"的语法,只有最适合团队的语法。
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
|
|
|
|
|
import { ref } from 'vue'
|
|
|
|
|
|
|
|
|
|
|
|
const selectedLang = ref('Python')
|
|
|
|
|
|
|
|
|
|
|
|
const languages = [
|
|
|
|
|
|
{ name: 'Python', icon: '🐍' },
|
|
|
|
|
|
{ name: 'Go', icon: '🐹' },
|
|
|
|
|
|
{ name: 'Node.js', icon: '💚' },
|
|
|
|
|
|
{ name: 'Java', icon: '☕' },
|
2026-02-13 22:10:03 +08:00
|
|
|
|
{ name: 'Rust', icon: '🦀' }
|
2026-01-18 12:21:49 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
const codes = {
|
|
|
|
|
|
Python: {
|
|
|
|
|
|
code: `print("Hello, World!")`,
|
|
|
|
|
|
filename: 'hello.py',
|
|
|
|
|
|
complexity: '极简'
|
|
|
|
|
|
},
|
|
|
|
|
|
Go: {
|
|
|
|
|
|
code: `package main
|
|
|
|
|
|
|
|
|
|
|
|
import "fmt"
|
|
|
|
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
|
|
fmt.Println("Hello, World!")
|
|
|
|
|
|
}`,
|
|
|
|
|
|
filename: 'hello.go',
|
|
|
|
|
|
complexity: '简洁'
|
|
|
|
|
|
},
|
|
|
|
|
|
'Node.js': {
|
|
|
|
|
|
code: `console.log("Hello, World!");`,
|
|
|
|
|
|
filename: 'hello.js',
|
|
|
|
|
|
complexity: '极简'
|
|
|
|
|
|
},
|
|
|
|
|
|
Java: {
|
|
|
|
|
|
code: `public class HelloWorld {
|
|
|
|
|
|
public static void main(String[] args) {
|
|
|
|
|
|
System.out.println("Hello, World!");
|
|
|
|
|
|
}
|
|
|
|
|
|
}`,
|
|
|
|
|
|
filename: 'HelloWorld.java',
|
|
|
|
|
|
complexity: '冗长'
|
|
|
|
|
|
},
|
|
|
|
|
|
Rust: {
|
|
|
|
|
|
code: `fn main() {
|
|
|
|
|
|
println!("Hello, World!");
|
|
|
|
|
|
}`,
|
|
|
|
|
|
filename: 'main.rs',
|
|
|
|
|
|
complexity: '简洁'
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const getCode = (lang) => {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
return codes[lang]
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const getLineCount = (lang) => {
|
|
|
|
|
|
return codes[lang].code.split('\n').length
|
|
|
|
|
|
}
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.syntax-comparison-demo {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
background: var(--vp-c-bg-soft);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
margin: 0.5rem 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.demo-header .icon {
|
|
|
|
|
|
font-size: 1.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.demo-header .title {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 1rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header .subtitle {
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
margin-left: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.intro-text {
|
|
|
|
|
|
font-size: 0.9rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
2026-02-13 22:10:03 +08:00
|
|
|
|
line-height: 1.6;
|
|
|
|
|
|
margin-bottom: 1rem;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.intro-text .highlight {
|
|
|
|
|
|
color: var(--vp-c-brand-1);
|
|
|
|
|
|
font-weight: 500;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.language-tabs {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
|
|
|
|
|
overflow-x: auto;
|
|
|
|
|
|
padding-bottom: 0.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.lang-tab {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 0.25rem;
|
|
|
|
|
|
padding: 0.5rem 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
background: var(--vp-c-bg);
|
2026-02-13 22:10:03 +08:00
|
|
|
|
border: 2px solid transparent;
|
|
|
|
|
|
border-radius: 6px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
cursor: pointer;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
transition: all 0.2s ease;
|
|
|
|
|
|
font-size: 0.85rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.lang-tab:hover {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.lang-tab.active {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.tab-icon {
|
|
|
|
|
|
font-size: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.code-display {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.code-window {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
background: #1e1e1e;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
overflow: hidden;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.window-header {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
padding: 0.5rem 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
background: #2d2d2d;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 0.5rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.window-controls {
|
|
|
|
|
|
display: flex;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 4px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.control {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
width: 10px;
|
|
|
|
|
|
height: 10px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.control.red {
|
|
|
|
|
|
background: #ff5f56;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.control.yellow {
|
|
|
|
|
|
background: #ffbd2e;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.control.green {
|
|
|
|
|
|
background: #27c93f;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.file-name {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
flex: 1;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
color: #858585;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-size: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.code-content {
|
|
|
|
|
|
margin: 0;
|
2026-02-14 20:23:34 +08:00
|
|
|
|
padding: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
background: #1e1e1e;
|
|
|
|
|
|
color: #d4d4d4;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-family: 'Monaco', 'Menlo', monospace;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
line-height: 1.5;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
overflow-x: auto;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.code-stats {
|
|
|
|
|
|
display: flex;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-item {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-label {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
margin-right: 0.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-value {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
color: var(--vp-c-brand-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.fade-enter-active,
|
|
|
|
|
|
.fade-leave-active {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
transition: opacity 0.2s ease;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.fade-enter-from,
|
|
|
|
|
|
.fade-leave-to {
|
|
|
|
|
|
opacity: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.info-box {
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.info-box .icon {
|
|
|
|
|
|
margin-right: 0.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
</style>
|