f35cddeb8b
## 新增组件 (14个) - CodeSmellDemo.vue: 代码异味识别演示 - DecisionMatrixDemo.vue: 决策矩阵工具 - DesignPatternCatalogDemo.vue: 设计模式目录 - DocStructureDemo.vue: 文档结构示例 - LicenseComparisonDemo.vue: 开源许可证对比 - OpenSourceWorkflowDemo.vue: 开源协作流程 - PatternPlaygroundDemo.vue: 设计模式演练场 - RefactoringDemo.vue: 重构实战演示 - SecurityChecklistDemo.vue: 安全检查清单 - TDDCycleDemo.vue: TDD 循环演示 - TechRadarDemo.vue: 技术雷达图 - TechWritingPracticeDemo.vue: 技术写作实践 - TestPyramidDemo.vue: 测试金字塔 - WebSecurityDemo.vue: Web 安全演示 ## 文档更新 (7篇) - code-quality-refactoring.md: 代码质量与重构 - design-patterns.md: 设计模式 - open-source-collaboration.md: 开源协作 - security-thinking.md: 安全思维 - technical-writing.md: 技术写作 - technology-selection.md: 技术选型 - testing-strategies.md: 测试策略 ## 其他变更 - 将 browser-as-os.md 内容合并到 computer-networks.md - 更新 .gitignore 和 theme/index.js
235 lines
6.2 KiB
Vue
235 lines
6.2 KiB
Vue
<template>
|
||
<div class="lc-root">
|
||
<h4 class="lc-title">开源许可证对比工具</h4>
|
||
|
||
<!-- Filter -->
|
||
<div class="lc-filter">
|
||
<span class="lc-filter-label">我的需求:</span>
|
||
<button
|
||
v-for="f in filters"
|
||
:key="f.id"
|
||
:class="['lc-tag', { 'lc-tag--on': activeFilters.includes(f.id) }]"
|
||
@click="toggle(f.id)"
|
||
>{{ f.label }}</button>
|
||
<button v-if="activeFilters.length" class="lc-tag lc-tag--clear" @click="activeFilters = []">清除筛选</button>
|
||
</div>
|
||
|
||
<!-- Recommendation -->
|
||
<div v-if="recommended" class="lc-recommend">
|
||
推荐许可证:<strong>{{ recommended.name }}</strong> — {{ recommended.summary }}
|
||
</div>
|
||
|
||
<!-- Table -->
|
||
<div class="lc-table-wrap">
|
||
<table class="lc-table">
|
||
<thead>
|
||
<tr>
|
||
<th>许可证</th>
|
||
<th v-for="p in permissions" :key="p.id">{{ p.label }}</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
<tr
|
||
v-for="l in licenses"
|
||
:key="l.id"
|
||
:class="{ 'lc-row--hl': recommended && recommended.id === l.id }"
|
||
>
|
||
<td class="lc-name-cell">
|
||
<strong>{{ l.name }}</strong>
|
||
<span class="lc-desc">{{ l.summary }}</span>
|
||
</td>
|
||
<td v-for="p in permissions" :key="p.id" class="lc-cell">
|
||
<span v-if="l.perms[p.id] === true" class="lc-yes">✓</span>
|
||
<span v-else-if="l.perms[p.id] === false" class="lc-no">✗</span>
|
||
<span v-else class="lc-cond">⚠</span>
|
||
</td>
|
||
</tr>
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
|
||
<!-- Legend -->
|
||
<div class="lc-legend">
|
||
<span><span class="lc-yes">✓</span> 允许</span>
|
||
<span><span class="lc-no">✗</span> 不允许/限制</span>
|
||
<span><span class="lc-cond">⚠</span> 有条件</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref, computed } from 'vue'
|
||
|
||
const permissions = [
|
||
{ id: 'commercial', label: '商用' },
|
||
{ id: 'modify', label: '修改' },
|
||
{ id: 'distribute', label: '分发' },
|
||
{ id: 'patent', label: '专利授权' },
|
||
{ id: 'private', label: '私用' },
|
||
{ id: 'copyleft', label: '需开源衍生' },
|
||
{ id: 'liability', label: '免责' }
|
||
]
|
||
|
||
const licenses = [
|
||
{
|
||
id: 'mit', name: 'MIT', summary: '最宽松,几乎无限制',
|
||
perms: { commercial: true, modify: true, distribute: true, patent: false, private: true, copyleft: false, liability: true },
|
||
tags: ['commercial', 'simple', 'private']
|
||
},
|
||
{
|
||
id: 'apache2', name: 'Apache 2.0', summary: '宽松 + 专利保护',
|
||
perms: { commercial: true, modify: true, distribute: true, patent: true, private: true, copyleft: false, liability: true },
|
||
tags: ['commercial', 'patent', 'private']
|
||
},
|
||
{
|
||
id: 'gpl3', name: 'GPL 3.0', summary: '强 Copyleft,衍生必须开源',
|
||
perms: { commercial: true, modify: true, distribute: true, patent: true, private: true, copyleft: true, liability: true },
|
||
tags: ['copyleft', 'patent']
|
||
},
|
||
{
|
||
id: 'bsd2', name: 'BSD 2-Clause', summary: '类似 MIT,极简宽松',
|
||
perms: { commercial: true, modify: true, distribute: true, patent: false, private: true, copyleft: false, liability: true },
|
||
tags: ['commercial', 'simple', 'private']
|
||
},
|
||
{
|
||
id: 'mpl2', name: 'MPL 2.0', summary: '文件级 Copyleft,折中方案',
|
||
perms: { commercial: true, modify: true, distribute: true, patent: true, private: true, copyleft: 'cond', liability: true },
|
||
tags: ['commercial', 'patent', 'copyleft']
|
||
}
|
||
]
|
||
|
||
const filters = [
|
||
{ id: 'commercial', label: '允许商用' },
|
||
{ id: 'patent', label: '需要专利保护' },
|
||
{ id: 'simple', label: '尽量简单' },
|
||
{ id: 'copyleft', label: '要求衍生开源' },
|
||
{ id: 'private', label: '允许闭源使用' }
|
||
]
|
||
|
||
const activeFilters = ref([])
|
||
|
||
function toggle(id) {
|
||
const idx = activeFilters.value.indexOf(id)
|
||
if (idx >= 0) activeFilters.value.splice(idx, 1)
|
||
else activeFilters.value.push(id)
|
||
}
|
||
|
||
const recommended = computed(() => {
|
||
if (!activeFilters.value.length) return null
|
||
let best = null
|
||
let bestScore = -1
|
||
for (const l of licenses) {
|
||
const score = activeFilters.value.filter(f => l.tags.includes(f)).length
|
||
if (score > bestScore) { bestScore = score; best = l }
|
||
}
|
||
return bestScore > 0 ? best : null
|
||
})
|
||
</script>
|
||
|
||
<style scoped>
|
||
.lc-root {
|
||
margin: 1.5em 0;
|
||
padding: 1.2em;
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 12px;
|
||
background: var(--vp-c-bg-soft);
|
||
}
|
||
.lc-title {
|
||
margin: 0 0 1em;
|
||
font-size: 1.05em;
|
||
font-weight: 600;
|
||
text-align: center;
|
||
}
|
||
|
||
/* Filter */
|
||
.lc-filter {
|
||
display: flex;
|
||
flex-wrap: wrap;
|
||
align-items: center;
|
||
gap: 8px;
|
||
margin-bottom: 1em;
|
||
}
|
||
.lc-filter-label {
|
||
font-size: 0.9em;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
.lc-tag {
|
||
padding: 4px 12px;
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 16px;
|
||
background: var(--vp-c-bg);
|
||
color: var(--vp-c-text-2);
|
||
font-size: 0.82em;
|
||
cursor: pointer;
|
||
transition: all 0.2s;
|
||
}
|
||
.lc-tag:hover { border-color: var(--vp-c-brand-1); }
|
||
.lc-tag--on {
|
||
background: var(--vp-c-brand-1);
|
||
border-color: var(--vp-c-brand-1);
|
||
color: #fff;
|
||
}
|
||
.lc-tag--clear {
|
||
color: var(--vp-c-text-3);
|
||
font-size: 0.78em;
|
||
}
|
||
|
||
/* Recommend */
|
||
.lc-recommend {
|
||
padding: 0.6em 1em;
|
||
margin-bottom: 1em;
|
||
border-radius: 8px;
|
||
background: var(--vp-c-brand-soft);
|
||
font-size: 0.9em;
|
||
color: var(--vp-c-brand-1);
|
||
}
|
||
|
||
/* Table */
|
||
.lc-table-wrap {
|
||
overflow-x: auto;
|
||
margin-bottom: 0.8em;
|
||
}
|
||
.lc-table {
|
||
width: 100%;
|
||
border-collapse: collapse;
|
||
font-size: 0.88em;
|
||
}
|
||
.lc-table th,
|
||
.lc-table td {
|
||
padding: 8px 10px;
|
||
border: 1px solid var(--vp-c-divider);
|
||
text-align: center;
|
||
}
|
||
.lc-table th {
|
||
background: var(--vp-c-bg);
|
||
font-weight: 600;
|
||
font-size: 0.85em;
|
||
white-space: nowrap;
|
||
}
|
||
.lc-name-cell {
|
||
text-align: left !important;
|
||
min-width: 130px;
|
||
}
|
||
.lc-desc {
|
||
display: block;
|
||
font-size: 0.8em;
|
||
color: var(--vp-c-text-3);
|
||
font-weight: 400;
|
||
}
|
||
.lc-row--hl {
|
||
background: var(--vp-c-brand-soft);
|
||
}
|
||
.lc-cell { font-size: 1.1em; }
|
||
.lc-yes { color: #10b981; font-weight: 700; }
|
||
.lc-no { color: #ef4444; font-weight: 700; }
|
||
.lc-cond { color: #f59e0b; }
|
||
|
||
/* Legend */
|
||
.lc-legend {
|
||
display: flex;
|
||
gap: 1.5em;
|
||
font-size: 0.8em;
|
||
color: var(--vp-c-text-3);
|
||
}
|
||
</style>
|