API

在SpreadJS中,您可以使用全面的API以编程方式管理线程注释。该API允许您创建、检索、更新和删除线程注释及其回复,以及管理它们的解决状态。

管理线程注释 线程注释提供了一系列API。您可以通过编程方式创建、管理和组织线程注释及回复。 创建线程注释 要在单元格上创建线程注释,请使用工作表的threadedComments管理器上的add方法。然后添加线程的第一个回复: 获取和删除线程注释 您可以使用get方法获取线程注释,并使用remove方法删除它: 管理已解决状态 您可以使用resolved方法将线程注释标记为已解决或未解决: 添加回复 使用add方法向线程注释添加回复。每个回复可以包含文本、提及和链接: 更新回复 使用set方法更新现有回复: 删除回复 使用remove方法删除特定回复: 用户管理 配置用户管理器 在使用线程注释之前,请使用用户查找和搜索函数配置UserManager: 设置当前用户 设置将创作新评论和回复的当前用户:
window.onload = function () { var spread = new GC.Spread.Sheets.Workbook(document.getElementById("ss")); initSpread(spread); }; // User IDs in GUID format var USER_IDS = { ALICE: "{715CFD19-2BB9-4B94-DE0C-08D0F9A019DB}", BOB: "{8A3E2F47-5C1D-4E8B-A9F2-3B7C6D8E9F01}", CAROL: "{C4D5E6F7-8A9B-0C1D-2E3F-4A5B6C7D8E9F}", DAVID: "{D1E2F3A4-B5C6-7D8E-9F0A-1B2C3D4E5F67}" }; var activeThreadedComment = null; var activeReplyIndex = null; function initSpread(spread) { // Configure UserManager first configureUserManager(); spread.suspendPaint(); var sheet = spread.getActiveSheet(); sheet.options.allowCellOverflow = true; // Create a sample data table createSampleTable(sheet); // Add threaded comments to demonstrate various APIs addThreadedComments(sheet); spread.resumePaint(); // Bind selection change event var spreadNS = GC.Spread.Sheets; spread.bind(spreadNS.Events.SelectionChanged, function (e, info) { var sheetTmp = info.sheet; var row = sheetTmp.getActiveRowIndex(); var col = sheetTmp.getActiveColumnIndex(); var threadedComment = sheetTmp.threadedComments.get(row, col); if (threadedComment) { _getElementById("commentTip").innerHTML = "Threaded Comment in Cell [ " + row + " : " + col + " ]"; activeThreadedComment = threadedComment; activeReplyIndex = null; loadReplies(); // Automatically load replies } else { _getElementById("commentTip").innerHTML = "No Threaded Comment"; activeThreadedComment = null; activeReplyIndex = null; clearReplies(); // Clear replies list } updateLabels(spread); }); // Bind button events bindButtonEvents(spread); } function configureUserManager() { // Mock user database with GUID-style IDs and avatars var users = [ { id: USER_IDS.ALICE, name: "Alice Johnson", email: "alice.johnson@company.com", avatar: { kind: "data", dataUrl: "" } }, { id: USER_IDS.BOB, name: "Bob Smith", email: "bob.smith@company.com", avatar: { kind: "data", dataUrl: "" } }, { id: USER_IDS.CAROL, name: "Carol Manager", email: "carol.manager@company.com", avatar: { kind: "data", dataUrl: "" } }, { id: USER_IDS.DAVID, name: "David Accountant", email: "david.accountant@company.com", avatar: { kind: "data", dataUrl: "" } } ]; GC.Spread.Common.UserManager.configure({ get: async (userId) => { if (userId === undefined) { return; } return new Promise((resolve) => { const user = users.find(u => u.id === userId); resolve(user); }); }, search: async (query) => { return new Promise((resolve) => { resolve(users.filter(u => u.name.toLowerCase().includes(query.toLowerCase()) || u.email.toLowerCase().includes(query.toLowerCase()) )); }); } }); // Set current user GC.Spread.Common.UserManager.current(USER_IDS.ALICE); } function addThreadedComments(sheet) { var threadedCommentManager = sheet.threadedComments; // Example 1: Simple threaded comment with one reply on Reserve var tc1 = threadedCommentManager.add(9, 2); tc1.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "Please verify if 10% reserve is sufficient for this quarter." } ], authorId: USER_IDS.ALICE, createdAt: new Date() }); // Example 2: Threaded comment with multiple replies on Total Budget var tc2 = threadedCommentManager.add(10, 2); tc2.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "The total budget looks reasonable." } ], authorId: USER_IDS.ALICE, createdAt: new Date(Date.now() - 86400000) // 1 day ago }); tc2.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "Confirmed. Ready for approval." } ], authorId: USER_IDS.BOB, createdAt: new Date(Date.now() - 43200000) // 12 hours ago }); tc2.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "Approved!" } ], authorId: USER_IDS.CAROL, createdAt: new Date() }); // Example 3: Resolved threaded comment on Operations var tc3 = threadedCommentManager.add(4, 2); tc3.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "Should we increase the operations budget?" } ], authorId: USER_IDS.BOB, createdAt: new Date(Date.now() - 172800000) // 2 days ago }); tc3.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "No, this amount aligns with our Q3 plans." } ], authorId: USER_IDS.ALICE, createdAt: new Date(Date.now() - 86400000) // 1 day ago }); // Mark this thread as resolved tc3.resolved(true); // Example 4: Comment with mention and link on Development variance var tc4 = threadedCommentManager.add(3, 4); tc4.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: "Hey " }, { type: GC.Spread.Sheets.ThreadedComments.ContentType.mention, userId: USER_IDS.DAVID }, { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: ", please review the budget variance in Development department: " }, { type: GC.Spread.Sheets.ThreadedComments.ContentType.link, href: "https://example.com/budget-policy", text: "Budget Policy" } ], authorId: USER_IDS.ALICE, createdAt: new Date() }); } function createSampleTable(sheet) { // Set column widths sheet.setColumnWidth(1, 150); sheet.setColumnWidth(2, 120); sheet.setColumnWidth(3, 100); sheet.setColumnWidth(4, 120); // Headers sheet.setValue(1, 1, "Department"); sheet.setValue(1, 2, "Q3 Budget"); sheet.setValue(1, 3, "Q3 Actual"); sheet.setValue(1, 4, "Variance"); sheet.getRange(1, 1, 1, 4).font("bold 12pt Arial"); // Department budget data sheet.setValue(2, 1, "Marketing"); sheet.setValue(2, 2, 15000); sheet.setValue(2, 3, 14200); sheet.getCell(2, 4).formula("=C3-D3"); sheet.setValue(3, 1, "Development"); sheet.setValue(3, 2, 25000); sheet.setValue(3, 3, 26800); sheet.getCell(3, 4).formula("=C4-D4"); sheet.setValue(4, 1, "Operations"); sheet.setValue(4, 2, 10000); sheet.setValue(4, 3, 9850); sheet.getCell(4, 4).formula("=C5-D5"); sheet.setValue(5, 1, "Sales"); sheet.setValue(5, 2, 18000); sheet.setValue(5, 3, 17500); sheet.getCell(5, 4).formula("=C6-D6"); sheet.setValue(6, 1, "HR"); sheet.setValue(6, 2, 8000); sheet.setValue(6, 3, 8200); sheet.getCell(6, 4).formula("=C7-D7"); sheet.setValue(7, 1, "IT Support"); sheet.setValue(7, 2, 12000); sheet.setValue(7, 3, 11600); sheet.getCell(7, 4).formula("=C8-D8"); // Subtotal row sheet.setValue(8, 1, "Subtotal"); sheet.getCell(8, 2).formula("=SUM(C3:C8)"); sheet.getCell(8, 3).formula("=SUM(D3:D8)"); sheet.getCell(8, 4).formula("=SUM(E3:E8)"); sheet.getRange(8, 1, 1, 4).font("bold 11pt Arial"); // Reserve calculation sheet.setValue(9, 1, "Reserve (10%)"); sheet.getCell(9, 2).formula("=C8*0.1"); sheet.setValue(9, 3, ""); sheet.setValue(9, 4, ""); sheet.getCell(9, 1).font("bold 11pt Arial"); sheet.getCell(9, 2).font("bold 11pt Arial"); sheet.getCell(9, 2).backColor("#FFEB9C"); sheet.setValue(10, 1, "Total Budget"); sheet.getCell(10, 2).formula("=C8+C9"); sheet.getCell(10, 3).formula("=D8"); sheet.getCell(10, 4).formula("=C11-D11"); sheet.getRange(10, 1, 1, 4).font("bold 12pt Arial"); sheet.getRange(10, 2, 1, 3).backColor("#C6EFCE"); sheet.getRange(2, 2, 9, 3).formatter("$#,##0.00;[Red]-$#,##0.00"); } async function loadReplies() { if (activeThreadedComment) { var replies = activeThreadedComment.all(); var repliesList = _getElementById("repliesList"); repliesList.innerHTML = ""; if (replies && replies.length > 0) { for (var index = 0; index < replies.length; index++) { var reply = replies[index]; var option = document.createElement("option"); option.value = index; var message = ""; if (reply.message && reply.message.length > 0) { var messageParts = []; for (var i = 0; i < reply.message.length; i++) { var part = reply.message[i]; if (part.type === GC.Spread.Sheets.ThreadedComments.ContentType.text) { messageParts.push(part.value); } else if (part.type === GC.Spread.Sheets.ThreadedComments.ContentType.mention) { var user = await GC.Spread.Common.UserManager.get(part.userId); messageParts.push("@" + (user ? user.name : part.userId)); } else if (part.type === GC.Spread.Sheets.ThreadedComments.ContentType.link) { messageParts.push(part.text || part.href); } } message = messageParts.join(""); } option.text = "Reply " + (index + 1) + ": " + message.substring(0, 30) + (message.length > 30 ? "..." : ""); repliesList.appendChild(option); } } else { var option = document.createElement("option"); option.text = "No replies"; repliesList.appendChild(option); } } } function clearReplies() { var repliesList = _getElementById("repliesList"); repliesList.innerHTML = ""; var option = document.createElement("option"); option.text = "No replies loaded"; repliesList.appendChild(option); } function bindButtonEvents(spread) { // Select reply _getElementById("repliesList").addEventListener('change', function () { if (activeThreadedComment && this.value !== "") { activeReplyIndex = parseInt(this.value); updateLabels(spread); } }); // Add new reply _getElementById("btnAddReply").addEventListener('click', function () { if (activeThreadedComment) { var message = _getElementById("txtNewReply").value; if (message) { activeThreadedComment.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: message } ], authorId: GC.Spread.Common.UserManager.current(), createdAt: new Date() }); _getElementById("txtNewReply").value = ""; loadReplies(); updateLabels(spread); } } }); // Delete reply _getElementById("btnDeleteReply").addEventListener('click', function () { if (activeThreadedComment && activeReplyIndex !== null) { var replies = activeThreadedComment.all(); var replyToDelete = replies[activeReplyIndex]; if (replyToDelete) { activeThreadedComment.remove(activeReplyIndex); activeReplyIndex = null; loadReplies(); updateLabels(spread); } } }); // Toggle resolved status _getElementById("btnToggleResolved").addEventListener('click', function () { if (activeThreadedComment) { var currentStatus = activeThreadedComment.resolved(); activeThreadedComment.resolved(!currentStatus); updateLabels(spread); } }); // Delete entire thread _getElementById("btnDeleteThread").addEventListener('click', function () { if (activeThreadedComment) { var sheet = spread.getActiveSheet(); var row = sheet.getActiveRowIndex(); var col = sheet.getActiveColumnIndex(); sheet.threadedComments.remove(row, col); activeThreadedComment = null; activeReplyIndex = null; updateLabels(spread); clearReplies(); } }); // Add new thread _getElementById("btnAddThread").addEventListener('click', function () { var sheet = spread.getActiveSheet(); var row = sheet.getActiveRowIndex(); var col = sheet.getActiveColumnIndex(); var message = _getElementById("txtNewThread").value; if (message) { var existingComment = sheet.threadedComments.get(row, col); if (existingComment) { sheet.threadedComments.remove(row, col); } var tc = sheet.threadedComments.add(row, col); tc.add({ message: [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: message } ], authorId: GC.Spread.Common.UserManager.current(), createdAt: new Date() }); _getElementById("txtNewThread").value = ""; activeThreadedComment = tc; updateLabels(spread); } }); // Update reply message _getElementById("btnUpdateReply").addEventListener('click', function () { if (activeThreadedComment && activeReplyIndex !== null) { var newMessage = _getElementById("txtUpdateMessage").value; if (newMessage) { var replies = activeThreadedComment.all(); var replyToUpdate = replies[activeReplyIndex]; if (replyToUpdate) { replyToUpdate.message = [ { type: GC.Spread.Sheets.ThreadedComments.ContentType.text, value: newMessage } ]; replyToUpdate.modifiedAt = new Date(); activeThreadedComment.set(activeReplyIndex, replyToUpdate); _getElementById("txtUpdateMessage").value = ""; loadReplies(); updateLabels(spread); } } } }); // Change current user _getElementById("comboCurrentUser").addEventListener('change', function () { var userId = this.value; GC.Spread.Common.UserManager.current(userId); }); } async function updateLabels(spread) { // Clear all fields first _getElementById("lblThreadLocation").innerHTML = "-"; _getElementById("lblThreadResolved").innerHTML = "-"; _getElementById("lblThreadRepliesCount").innerHTML = "-"; _getElementById("lblReplyAuthor").innerHTML = "-"; _getElementById("lblReplyCreatedAt").innerHTML = "-"; _getElementById("lblReplyMessage").innerHTML = "-"; if (activeThreadedComment) { _getElementById("lblThreadLocation").innerHTML = "Row: " + activeThreadedComment.row() + ", Col: " + activeThreadedComment.col(); _getElementById("lblThreadResolved").innerHTML = activeThreadedComment.resolved() ? "Yes" : "No"; var replies = activeThreadedComment.all(); _getElementById("lblThreadRepliesCount").innerHTML = replies.length.toString(); if (activeReplyIndex !== null) { var activeReply = replies[activeReplyIndex]; if (activeReply) { // Get author name from UserManager var authorUser = await GC.Spread.Common.UserManager.get(activeReply.authorId); _getElementById("lblReplyAuthor").innerHTML = authorUser ? authorUser.name : (activeReply.authorId || "-"); _getElementById("lblReplyCreatedAt").innerHTML = activeReply.createdAt ? activeReply.createdAt.toLocaleString() : "-"; if (activeReply.message && activeReply.message.length > 0) { var messageParts = []; for (var i = 0; i < activeReply.message.length; i++) { var part = activeReply.message[i]; if (part.type === GC.Spread.Sheets.ThreadedComments.ContentType.text) { messageParts.push(part.value); } else if (part.type === GC.Spread.Sheets.ThreadedComments.ContentType.mention) { var user = await GC.Spread.Common.UserManager.get(part.userId); messageParts.push("@" + (user ? user.name : part.userId)); } else if (part.type === GC.Spread.Sheets.ThreadedComments.ContentType.link) { messageParts.push('<a href="' + part.href + '" target="_blank">' + (part.text || part.href) + '</a>'); } } _getElementById("lblReplyMessage").innerHTML = messageParts.join(""); } } } } spread.refresh(); } function _getElementById(id) { return document.getElementById(id); }
<!doctype html> <html style="height:100%;font-size:14px;"> <head> <meta name="spreadjs culture" content="zh-cn" /> <meta charset="utf-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <link rel="stylesheet" type="text/css" href="$DEMOROOT$/zh/purejs/node_modules/@grapecity-software/spread-sheets/styles/gc.spread.sheets.excel2013white.css"> <script src="$DEMOROOT$/zh/purejs/node_modules/@grapecity-software/spread-sheets/dist/gc.spread.sheets.all.min.js" type="text/javascript"></script> <script src="$DEMOROOT$/zh/purejs/node_modules/@grapecity-software/spread-sheets-resources-zh/dist/gc.spread.sheets.resources.zh.min.js" type="text/javascript"></script> <script src="$DEMOROOT$/spread/source/js/license.js" type="text/javascript"></script> <script src="app.js" type="text/javascript"></script> <link rel="stylesheet" type="text/css" href="styles.css"> </head> <body> <div class="tc-sample-tutorial"> <div id="ss" class="tc-sample-spreadsheets"></div> <div class="tc-options-container"> <div class="tc-option-row"> <h4 class="tc-title">选择带有线程批注的单元格: <span id="commentTip" class="tc-comment-tip">无线程批注</span> </h4> </div> <!-- 当前用户选择 --> <div class="tc-section"> <h5 class="tc-section-title">当前用户</h5> <div class="tc-option-row"> <label class="tc-label">切换用户:</label> <select id="comboCurrentUser" class="tc-select" style="width: 100%;"> <option value="{715CFD19-2BB9-4B94-DE0C-08D0F9A019DB}">Alice Johnson</option> <option value="{8A3E2F47-5C1D-4E8B-A9F2-3B7C6D8E9F01}">Bob Smith</option> <option value="{C4D5E6F7-8A9B-0C1D-2E3F-4A5B6C7D8E9F}">Carol Manager</option> <option value="{D1E2F3A4-B5C6-7D8E-9F0A-1B2C3D4E5F67}">David Accountant</option> </select> </div> </div> <!-- 线程信息 --> <div class="tc-section"> <h5 class="tc-section-title">线程信息</h5> <div class="tc-info-row"> <label class="tc-label">位置:</label> <span id="lblThreadLocation" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">已解决:</label> <span id="lblThreadResolved" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">回复数量:</label> <span id="lblThreadRepliesCount" class="tc-info-value">-</span> </div> </div> <!-- 线程操作 --> <div class="tc-section"> <h5 class="tc-section-title">线程操作</h5> <div class="tc-option-row"> <input type="text" id="txtNewThread" class="tc-input" placeholder="输入线程消息..." style="width: 100%; margin-bottom: 6px;" /> <button id="btnAddThread" class="tc-btn tc-btn-primary">添加新线程</button> </div> <div class="tc-option-row"> <button id="btnToggleResolved" class="tc-btn tc-btn-secondary">切换解决状态</button> <button id="btnDeleteThread" class="tc-btn tc-btn-danger">删除线程</button> </div> </div> <!-- 回复管理 --> <div class="tc-section"> <h5 class="tc-section-title">回复管理</h5> <div class="tc-option-row"> <label class="tc-label">选择回复:</label> <select id="repliesList" class="tc-select" size="4" style="width: 100%; margin-top: 4px;"> <option>未加载回复</option> </select> </div> </div> <!-- 回复信息 --> <div class="tc-section"> <h5 class="tc-section-title">已选回复信息</h5> <div class="tc-info-row"> <label class="tc-label">作者:</label> <span id="lblReplyAuthor" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">创建时间:</label> <span id="lblReplyCreatedAt" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">消息:</label> <span id="lblReplyMessage" class="tc-info-value">-</span> </div> </div> <!-- 回复操作 --> <div class="tc-section"> <h5 class="tc-section-title">回复操作</h5> <div class="tc-option-row"> <input type="text" id="txtNewReply" class="tc-input" placeholder="输入回复消息..." style="width: 100%; margin-bottom: 6px;" /> <button id="btnAddReply" class="tc-btn tc-btn-primary">添加回复</button> </div> <div class="tc-option-row"> <input type="text" id="txtUpdateMessage" class="tc-input" placeholder="输入新消息..." style="width: 100%; margin-bottom: 6px;" /> <button id="btnUpdateReply" class="tc-btn tc-btn-secondary">更新已选回复</button> </div> <div class="tc-option-row"> <button id="btnDeleteReply" class="tc-btn tc-btn-danger">删除已选回复</button> </div> </div> </div> </div> </body> </html>
/* Container styles with specific prefixes */ .tc-sample-tutorial { position: relative; height: 100%; overflow: hidden; } .tc-sample-spreadsheets { width: calc(100% - 320px); height: 100%; overflow: hidden; float: left; } .tc-options-container { float: right; width: 320px; overflow: auto; padding: 12px; height: 100%; box-sizing: border-box; background: #fbfbfb; } /* Title styles */ .tc-title { margin: 0 0 12px 0; font-size: 15px; color: #333; } .tc-comment-tip { color: #007acc; font-weight: bold; } /* Section styles */ .tc-section { margin-bottom: 20px; padding: 12px; background: white; border: 1px solid #ddd; border-radius: 4px; } .tc-section-title { margin: 0 0 12px 0; padding-bottom: 8px; border-bottom: 2px solid #007acc; color: #333; font-size: 14px; font-weight: bold; } .tc-option-row { margin-bottom: 10px; } .tc-info-row { display: flex; align-items: center; margin-bottom: 8px; padding: 6px; background: #f9f9f9; border-radius: 3px; } .tc-info-row .tc-label { min-width: 100px; font-weight: bold; color: #555; display: inline-flex; align-items: center; margin: 0; } .tc-info-value { flex: 1; word-wrap: break-word; color: #333; display: inline-flex; align-items: center; min-height: 24px; } /* Form element styles */ .tc-label { display: inline-block; min-width: 100px; margin: 6px 0; font-weight: 500; color: #333; } .tc-input { padding: 6px 8px; box-sizing: border-box; border: 1px solid #ccc; border-radius: 3px; font-size: 13px; } .tc-select { padding: 6px 8px; box-sizing: border-box; border: 1px solid #ccc; border-radius: 3px; font-size: 13px; } /* Button styles */ .tc-btn { padding: 8px 12px; margin: 4px 4px 4px 0; border: none; border-radius: 3px; cursor: pointer; font-size: 13px; font-weight: 500; transition: background-color 0.2s; } .tc-btn-primary { background-color: #007acc; color: white; } .tc-btn-primary:hover { background-color: #005a9e; } .tc-btn-secondary { background-color: #5c6bc0; color: white; } .tc-btn-secondary:hover { background-color: #3f51b5; } .tc-btn-danger { background-color: #e53935; color: white; } .tc-btn-danger:hover { background-color: #c62828; } .tc-btn-info { background-color: #26a69a; color: white; } .tc-btn-info:hover { background-color: #00897b; } body { position: absolute; top: 0; bottom: 0; left: 0; right: 0; margin: 0; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, sans-serif; }