File size: 20,493 Bytes
e191679
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
<!DOCTYPE html>
<html lang="zh-TW">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>股票殖利率查詢 (HTML5 版)</title>
    <!-- Chosen Palette: Warm Neutrals -->
    <!-- Application Structure Plan: 
        1.  Header: 應用程式標題。
        2.  Search Area: 一個輸入框和搜尋按鈕,用於用戶輸入關鍵字。
        3.  History Area: 顯示在 localStorage 中儲存的搜尋記錄,可點擊回填。
        4.  Status Area: 顯示當前狀態 (載入中、載入完成、錯誤、搜尋結果)。
        5.  Results Area: 一個響應式表格 (Table),用於顯示從 Wespai 爬取並過濾後的資料。
        User Flow: 
        1.  頁面載入時,JS 自動觸發 `fetchData()`。
        2.  `fetchData()` 透過 CORS 代理抓取 Wespai 的 HTML。
        3.  `parseHTMLTable()` 解析 HTML 字串,將表格轉換為 JS 物件陣列 `allStockData`。
        4.  同時,`loadHistory()` 從 localStorage 讀取並顯示搜尋記錄。
        5.  用戶輸入關鍵字點擊 "搜尋"。
        6.  `performSearch()` 過濾 `allStockData` 陣列。
        7.  `renderTable()` 將過濾後的結果動態渲染到表格中。
        8.  `addHistory()` 將新關鍵字儲存到 localStorage 並更新顯示。
        Why this structure: 這是最直接的單頁應用 (SPA) 結構,將 "輸入" (搜尋框)、"狀態" (訊息欄) 和 "輸出" (表格) 清晰分離,並為 "搜尋記錄" 提供了專用區域,符合用戶的互動直覺。
    -->
    <!-- Visualization & Content Choices:
        -   Report Info: 來自 Wespai 的股票表格資料 (代號, 公司, 殖利率等)。
        -   Goal: 組織 (Organize) / 通知 (Inform)。
        -   Viz/Presentation Method: 互動式 HTML 表格。
        -   Interaction: 
            1.  Page Load: 透過 CORS 代理異步獲取 Wespai 原始 HTML。
            2.  JS Parsing: 將 HTML Table 轉換為 JS Array of Objects (allStockData)。
            3.  Keyword Filter: 用戶輸入關鍵字 (代號或公司名稱),JS 使用 `.filter()` 篩選 allStockData。
            4.  Dynamic Render: `renderTable()` 函數清空並重新填充 `<tbody>` 以顯示過濾後的結果。
            5.  History: `localStorage` 用於保存搜尋詞,並動態生成可點擊的 `<span>` 標籤以觸發新搜尋。
        -   Justification: 這是對原始 Python 腳本 (過濾 dataframe) 最直接的 JS 實現。表格是呈現此類數據的最佳方式。搜尋記錄增加了可用性。
        -   Library/Method: Vanilla JavaScript (DOMParser, fetch, localStorage), Tailwind CSS (UI)。
    -->
    <!-- CONFIRMATION: NO SVG graphics used. NO Mermaid JS used. -->
    <script src="https://cdn.tailwindcss.com"></script>
    <style>
        body {
            font-family: 'Inter', 'Noto Sans TC', sans-serif;
        }
        .clickable-tag {
            cursor: pointer;
            transition: all 0.2s;
        }
        .clickable-tag:hover {
            opacity: 0.7;
            transform: translateY(-1px);
        }
        /* 針對小螢幕優化表格字體 */
        @media (max-width: 640px) {
            table {
                font-size: 0.8rem;
            }
            .px-4 {
                padding-left: 0.75rem;
                padding-right: 0.75rem;
            }
            .py-2 {
                padding-top: 0.5rem;
                padding-bottom: 0.5rem;
            }
        }
    </style>
</head>
<body class="bg-gray-50 text-gray-800">

    <div class="container mx-auto p-4 md:p-8 max-w-full">
        <header class="mb-6">
            <h1 class="text-3xl font-bold text-gray-900">股票殖利率查詢 (HTML5 版)</h1>
            <p class="text-gray-600 mt-1">此工具會即時爬取 Wespai 資料並在您的瀏覽器中進行過濾。</p>
        </header>

        <!-- Search and History Section -->
        <div class="bg-white p-4 rounded-lg shadow-sm mb-6">
            <div class="flex flex-col sm:flex-row gap-2">
                <input type="text" id="search-input" class="flex-grow p-3 border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:outline-none" placeholder="輸入股票代號或公司名稱 (例如: 2330 或 台積)">
                <button id="search-button" class="bg-blue-600 text-white font-semibold py-3 px-6 rounded-lg shadow hover:bg-blue-700 transition duration-200">
                    搜尋
                </button>
            </div>
            <div id="history-container" class="mt-4 text-sm text-gray-600 flex flex-wrap gap-2 items-center">
                <strong class="mr-2">搜尋記錄:</strong>
                <!-- History tags will be injected here -->
            </div>
        </div>

        <!-- Status Message -->
        <div id="status-message" class="my-4 p-3 bg-blue-100 text-blue-800 rounded-lg text-center font-medium">
            正在初始化...
        </div>

        <!-- Results Table -->
        <div class="overflow-x-auto bg-white shadow-md rounded-lg">
            <table class="min-w-full divide-y divide-gray-200">
                <thead class="bg-gray-100">
                    <tr>
                        <!-- 這些欄位是根據你的 app.py 中的 TARGET_COLUMNS -->
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">代號</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">公司</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">配息</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">除息日</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">配股</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">除權日</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">股價</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">現金殖利率</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">殖利率</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">發息日</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">配息率</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">董監持股</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">3年平均股利</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">6年平均股利</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">10年平均股利</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">10年股利次數</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">1QEPS</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">2QEPS</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">3QEPS</th>
                        <th class="px-4 py-3 text-left text-xs font-bold text-gray-600 uppercase tracking-wider">今年累積EPS</th>
                    </tr>
                </thead>
                <tbody id="results-table-body" class="bg-white divide-y divide-gray-200">
                    <!-- Data rows will be injected here -->
                    <tr>
                        <td colspan="20" class="px-4 py-6 text-center text-gray-500">請稍候,正在載入資料...</td>
                    </tr>
                </tbody>
            </table>
        </div>

    </div>

    <script type="module">
        // -------------------------------------------------------------------------
        // 狀態和常數
        // -------------------------------------------------------------------------
        const WESPAI_URL = 'https://stock.wespai.com/rate114';
        const PROXY_URL = 'https://api.allorigins.win/raw?url='; // CORS 代理
        const HISTORY_KEY = 'stockSearchHistory';
        const MAX_HISTORY = 10;

        // 來自你的 app.py,我們將只解析和顯示這些欄位
        const TARGET_COLUMNS = [
            "代號", "公司", "配息", "除息日", "配股", "除權日", "股價",
            "現金殖利率", "殖利率", "發息日", "配息率", "董監持股",
            "3年平均股利", "6年平均股利", "10年平均股利", "10年股利次數",
            "1QEPS", "2QEPS", "3QEPS", "今年累積EPS",
        ];

        let allStockData = []; // 儲存所有解析後的股票資料
        let searchHistory = []; // 儲存搜尋記錄

        // -------------------------------------------------------------------------
        // DOM 元素
        // -------------------------------------------------------------------------
        const searchInput = document.getElementById('search-input');
        const searchButton = document.getElementById('search-button');
        const statusMessage = document.getElementById('status-message');
        const tableBody = document.getElementById('results-table-body');
        const historyContainer = document.getElementById('history-container');

        // -------------------------------------------------------------------------
        // 核心功能:資料抓取與解析
        // -------------------------------------------------------------------------

        /**
         * 從 Wespai 抓取 HTML 並解析成資料陣列
         */
        async function fetchData() {
            statusMessage.textContent = '正在從 Wespai 載入即時資料,請稍候...';
            statusMessage.className = 'my-4 p-3 bg-blue-100 text-blue-800 rounded-lg text-center font-medium';
            tableBody.innerHTML = `<tr><td colspan="${TARGET_COLUMNS.length}" class="px-4 py-6 text-center text-gray-500">正在載入...</td></tr>`;

            try {
                const response = await fetch(PROXY_URL + encodeURIComponent(WESPAI_URL));
                
                if (!response.ok) {
                    throw new Error(`HTTP 錯誤! 狀態: ${response.status}`);
                }

                const htmlString = await response.text();
                allStockData = parseHTMLTable(htmlString);

                if (allStockData.length === 0) {
                     statusMessage.textContent = '成功載入頁面,但無法解析到表格資料。';
                     statusMessage.className = 'my-4 p-3 bg-yellow-100 text-yellow-800 rounded-lg text-center font-medium';
                     tableBody.innerHTML = `<tr><td colspan="${TARGET_COLUMNS.length}" class="px-4 py-6 text-center text-gray-500">無法解析資料</td></tr>`;
                } else {
                    statusMessage.textContent = `資料載入完成。共 ${allStockData.length} 筆。請輸入關鍵字搜尋。`;
                    statusMessage.className = 'my-4 p-3 bg-green-100 text-green-800 rounded-lg text-center font-medium';
                    renderTable(allStockData); // 初始顯示所有資料
                }
            } catch (error) {
                console.error('抓取資料失敗:', error);
                statusMessage.textContent = '資料載入失敗。可能是 CORS 代理或目標網站暫時無法訪問。';
                statusMessage.className = 'my-4 p-3 bg-red-100 text-red-800 rounded-lg text-center font-medium';
                tableBody.innerHTML = `<tr><td colspan="${TARGET_COLUMNS.length}" class="px-4 py-6 text-center text-red-500">載入失敗</td></tr>`;
            }
        }

        /**
         * 解析 Wespai HTML 字串,提取表格資料
         * @param {string} htmlString - 包含表格的 HTML 原始碼
         * @returns {Array<object>} - 解析後的資料陣列
         */
        function parseHTMLTable(htmlString) {
            try {
                const parser = new DOMParser();
                const doc = parser.parseFromString(htmlString, 'text/html');
                const table = doc.querySelector('table.table-hover'); // 根據 Wespai 網站原始碼微調選擇器

                if (!table) {
                    console.error('在 HTML 中找不到 table.table-hover');
                    return [];
                }

                // 1. 獲取實際的表頭 (headers) 及其索引
                const actualHeaders = [...table.querySelectorAll('thead th')].map(th => th.textContent.trim());
                
                // 2. 建立目標欄位 (TARGET_COLUMNS) 到實際索引的映射
                const colIndices = {};
                TARGET_COLUMNS.forEach(targetCol => {
                    const index = actualHeaders.indexOf(targetCol);
                    colIndices[targetCol] = index; // index 為 -1 表示找不到
                });

                // 3. 遍歷資料列 (rows)
                const rows = [...table.querySelectorAll('tbody tr')];
                const parsedData = [];

                rows.forEach(row => {
                    const cells = [...row.querySelectorAll('td')];
                    const rowData = {};
                    let hasData = false;

                    TARGET_COLUMNS.forEach(targetCol => {
                        const index = colIndices[targetCol];
                        if (index !== -1 && cells[index]) {
                            rowData[targetCol] = cells[index].textContent.trim();
                            hasData = true;
                        } else {
                            rowData[targetCol] = ''; // 找不到對應欄位
                        }
                    });

                    if (hasData) {
                        parsedData.push(rowData);
                    }
                });

                return parsedData;
            } catch (error) {
                console.error('解析 HTML 表格失敗:', error);
                return [];
            }
        }

        // -------------------------------------------------------------------------
        // 核心功能:渲染與搜尋
        // -------------------------------------------------------------------------

        /**
         * 將資料渲染到 HTML 表格中
         * @param {Array<object>} data - 要顯示的股票資料
         */
        function renderTable(data) {
            tableBody.innerHTML = ''; // 清空舊資料

            if (data.length === 0) {
                tableBody.innerHTML = `<tr><td colspan="${TARGET_COLUMNS.length}" class="px-4 py-6 text-center text-gray-500">查無資料</td></tr>`;
                return;
            }

            const fragment = document.createDocumentFragment();
            data.forEach(stock => {
                const tr = document.createElement('tr');
                tr.className = 'hover:bg-gray-50';
                
                let cellsHTML = '';
                TARGET_COLUMNS.forEach(header => {
                    cellsHTML += `<td class="px-4 py-2 whitespace-nowrap text-sm text-gray-700">${stock[header] || ''}</td>`;
                });
                tr.innerHTML = cellsHTML;
                fragment.appendChild(tr);
            });

            tableBody.appendChild(fragment);
        }

        /**
         * 執行搜尋
         */
        function performSearch() {
            const keyword = searchInput.value.trim();

            if (!keyword) {
                // 如果關鍵字為空,顯示所有資料
                renderTable(allStockData);
                statusMessage.textContent = `顯示所有 ${allStockData.length} 筆資料。`;
                statusMessage.className = 'my-4 p-3 bg-green-100 text-green-800 rounded-lg text-center font-medium';
                return;
            }

            // 儲存搜尋記錄
            addHistory(keyword);

            // 執行過濾 (邏輯同你的 app.py)
            const lowerKeyword = keyword.toLowerCase();
            const filteredData = allStockData.filter(stock => {
                const nameMatch = stock['公司'] && stock['公司'].toLowerCase().includes(lowerKeyword);
                const codeMatch = stock['代號'] && stock['代號'].includes(keyword); // 代號通常是精確比對或
                return nameMatch || codeMatch;
            });

            // 渲染過濾後的表格
            renderTable(filteredData);

            // 更新狀態
            if (filteredData.length === 0) {
                statusMessage.textContent = `找不到關於 "${keyword}" 的資料。`;
                statusMessage.className = 'my-4 p-3 bg-yellow-100 text-yellow-800 rounded-lg text-center font-medium';
            } else {
                statusMessage.textContent = `找到 ${filteredData.length} 筆關於 "${keyword}" 的資料。`;
                statusMessage.className = 'my-4 p-3 bg-green-100 text-green-800 rounded-lg text-center font-medium';
            }
        }

        // -------------------------------------------------------------------------
        // 輔助功能:搜尋記錄 (LocalStorage)
        // -------------------------------------------------------------------------

        /**
         * 從 localStorage 載入搜尋記錄
         */
        function loadHistory() {
            const stored = localStorage.getItem(HISTORY_KEY);
            if (stored) {
                searchHistory = JSON.parse(stored);
                renderHistory();
            }
        }

        /**
         * 將新關鍵字添加到搜尋記錄
         * @param {string} keyword - 搜尋關鍵字
         */
        function addHistory(keyword) {
            // 移除已存在的相同記錄,確保新的在最前面
            const existingIndex = searchHistory.indexOf(keyword);
            if (existingIndex > -1) {
                searchHistory.splice(existingIndex, 1);
            }

            // 加到最前面
            searchHistory.unshift(keyword);

            // 限制記錄數量
            if (searchHistory.length > MAX_HISTORY) {
                searchHistory = searchHistory.slice(0, MAX_HISTORY);
            }

            // 儲存回 localStorage
            localStorage.setItem(HISTORY_KEY, JSON.stringify(searchHistory));
            
            // 重新渲染顯示
            renderHistory();
        }

        /**
         * 將搜尋記錄渲染到頁面上
         */
        function renderHistory() {
            // 清空舊標籤 (保留 "搜尋記錄:" 標題)
            historyContainer.innerHTML = '<strong class="mr-2">搜尋記錄:</strong>';
            
            searchHistory.forEach(term => {
                const tag = document.createElement('span');
                tag.textContent = term;
                tag.className = 'clickable-tag bg-gray-200 text-gray-700 px-3 py-1 rounded-full text-xs font-medium shadow-sm';
                
                // 點擊標籤時,觸發搜尋
                tag.onclick = () => {
                    searchInput.value = term;
                    performSearch();
                };
                historyContainer.appendChild(tag);
            });
        }

        // -------------------------------------------------------------------------
        // 初始化
        // -------------------------------------------------------------------------
        document.addEventListener('DOMContentLoaded', () => {
            // 綁定事件
            searchButton.addEventListener('click', performSearch);
            searchInput.addEventListener('keyup', (e) => {
                if (e.key === 'Enter') {
                    performSearch();
                }
            });

            // 載入初始資料
            loadHistory();
            fetchData();
        });

    </script>
</body>
</html>