${esc(title)}
${lead ? `${esc(lead)}
` : ''}// 學習教材 HTML 渲染:白話化、互動路徑、案例→通用原則
(function (global) {
const KIND_LABEL = {
overview: '課綱總覽', principleMap: '原則地圖', quiz: '練習題庫',
category: '學習分類', case: '案例講解', principle: '投資原則',
term: '名詞', company: '公司', episode: '單集',
};
const LEARN_PATHS = {
market: {
title: '我想知道:現在大環境適不適合加碼?',
lead: '先別猜明天漲跌。用利率、通膨、就業、信用四條線,決定你該積極還是保守。',
steps: [
{ title: '① 讀懂大環境在說什麼', body: '利率往哪走、通膨高不高、就業強不強——這三個決定「順風還是逆風」。', read: { kind: 'category', id: '總經與利率', label: '總經與利率(分類)' }, practice: { view: 'macro', label: '打開總經儀表板' } },
{ title: '② 用一套流程決定倉位', body: '不是滿倉或空手,而是「這週該偏進攻還是偏防守」。', read: { kind: 'case', id: '總經數據怎麼看', label: '案例:總經數據怎麼看' }, practice: { view: 'macro', label: '對照 CPI、非農卡片' } },
{ title: '③ 把原則記下來', body: '每次調倉都寫「為什麼」,避免事後用結果論合理化。', read: { kind: 'category', id: '交易與資金管理', label: '交易與資金管理' }, practice: { view: 'journal', label: '寫一筆復盤' } },
],
},
stock: {
title: '我想知道:這家公司值不值得研究?',
lead: '把財報、生意模式、估值、趨勢拆成檢查清單,重複用在新公司上。',
steps: [
{ title: '① 財報在說什麼', body: '營收、毛利、EPS、財測——先會讀,再談貴不貴。', read: { kind: 'category', id: '財報基本功', label: '財報基本功' }, read2: { kind: 'case', id: 'NVIDIA財報怎麼看', label: '案例:財報怎麼看' }, practice: { view: 'stock', label: '個股工具 · 財報健檢' } },
{ title: '② 生意好不好、有沒有護城河', body: '數字背後是定價權與產業位置,別只看熱門題材。', read: { kind: 'category', id: '護城河與商業模式', label: '護城河與商業模式' }, practice: { view: 'stock', label: '個股工具 · 投資地圖' } },
{ title: '③ 用案例練「可重複的判斷」', body: 'NVDA、台積電只是例子;重點是抽出你下次也能用的問題。', read: { kind: 'case', id: 'NVIDIA決策複盤', label: '案例:決策複盤(框架)' }, practice: { view: 'stock', label: '查一檔你關心的股票' } },
],
},
trade: {
title: '我想知道:這筆交易哪裡做對、哪裡做錯?',
lead: '賺賠是結果;真正要檢查的是當初的理由、風險與紀律有沒有成立。',
steps: [
{ title: '① 避開結果論', body: '賺了不代表判斷對,賠了也不代表一定錯——先分清楚。', read: { kind: 'principle', id: '原則九十六:結果論陷阱(Outcome Bias)', label: '原則:結果論陷阱' }, practice: { view: 'journal', label: '打開交易復盤' } },
{ title: '② 進出場要有依據', body: '進場理由、停損/減倉規則寫清楚,復盤才有東西可改。', read: { kind: 'category', id: '交易與資金管理', label: '交易與資金管理' }, practice: { view: 'journal', label: '新增一筆交易' } },
{ title: '③ 定期回頭對照原則', body: '把常犯的錯連回具體原則,下次遇到類似情境才改得動。', read: { kind: 'principleMap', id: '心法地圖', label: '原則地圖(分群索引)' }, practice: { view: 'learn', section: 'quiz', label: '做練習題' } },
],
},
};
const PRINCIPLE_GROUPS = [
{ id: 'macro', label: '大局與倉位', re: /降息|升息|通膨|總經|倉位|現金|信用|利率|PMI|非農|CPI|風險偏好/ },
{ id: 'research', label: '研究與選股', re: /Capex|毛利率|財報|估值|定價權|護城河|供給|營收|EPS|財測|產業/ },
{ id: 'trade', label: '交易與紀律', re: /減倉|停損|結果論|賣|買|趨勢|紀律|倉位|觸發|弱|強|復盤/ },
{ id: 'mind', label: '心態與認知', re: /新聞|情緒|認知|耐心|時間|概率|陷阱|偏誤/ },
];
function deEmmy(text) {
return String(text || '')
.replace(/Emmy 投資心法/g, '投資原則庫')
.replace(/Emmy 投資台/g, '投資學習台')
.replace(/\bEmmy\b/g, '講者')
.replace(/110 條原則/g, '完整原則庫')
.replace(/心法地圖/g, '原則地圖')
.replace(/心法/g, '原則');
}
function cleanPrincipleTitle(title) {
return deEmmy(String(title || ''))
.replace(/^原則[^::]+[::]\s*/, '')
.trim();
}
function extractWikiLinks(body) {
const links = [];
const re = /\[\[([^\]]+)\]\]/g;
let m;
while ((m = re.exec(String(body || '')))) links.push(m[1].trim());
return links;
}
function resolveLinkTarget(raw, linkMap) {
if (!raw || !linkMap) return null;
const key = raw.trim();
return linkMap[key] || linkMap[key.split('#').pop()] || linkMap[key.split('/').pop()] || null;
}
function extractPrinciples(note, linkMap, principles) {
const out = [];
const seen = new Set();
const add = (p) => {
if (!p || seen.has(p.id)) return;
seen.add(p.id);
out.push(p);
};
for (const raw of extractWikiLinks(note.body)) {
const hit = resolveLinkTarget(raw, linkMap);
if (hit && hit.kind === 'principle') {
const p = (principles || []).find(x => x.id === hit.id);
add(p || { id: hit.id, title: raw.split('#').pop() || raw });
}
if (/投資心法#|投資原則#|原則/.test(raw)) {
const id = raw.includes('#') ? raw.split('#').slice(1).join('#') : raw;
const p = (principles || []).find(x => x.id === id || x.title === id);
add(p || { id, title: id });
}
}
return out.slice(0, 12);
}
function leadFromNote(note) {
if (note.summary) return deEmmy(note.summary);
const body = deEmmy(note.body || '');
for (const line of body.split('\n')) {
let l = line.trim();
if (!l || /^#/.test(l)) continue;
if (l.startsWith('>')) l = l.replace(/^>\s?/, '');
l = l.replace(/\[\[([^\]|]+)(\|[^\]]+)?\]\]/g, '$1').replace(/[*`]/g, '').trim();
if (l.length > 12) return l.slice(0, 160);
}
return '';
}
function buildTocFromMarkdown(md) {
const items = [];
for (const line of String(md || '').split('\n')) {
const m = line.match(/^(#{2,3})\s+(.+)$/);
if (!m) continue;
const level = m[1].length;
const text = deEmmy(m[2].replace(/\[\[([^\]|]+)(\|([^\]]+))?\]\]/g, (_, a, _b, c) => c || a)).trim();
const id = 'sec-' + items.length;
items.push({ level, text, id });
}
return items.slice(0, 14);
}
function renderPathSteps(pathId, path) {
return path.steps.map((step, i) => {
const links = [
step.read ? `` : '',
step.read2 ? `` : '',
].filter(Boolean).join('');
const practice = step.practice
? ``
: '';
return ` ${deEmmy(step.body)}${i + 1}${deEmmy(step.title)}
不用從頭讀完。展開下面步驟,依序「讀一篇 → 連到工具 → 做一次判斷」。同一套問題可以反覆用在不同股票與不同月份。
這些原則不只適用這一檔公司——下次遇到類似情境,可以直接拿來問自己。
${esc(lead)}
` : ''}