API

In SpreadJS, you can manage threaded comments programmatically using the comprehensive API. The API allows you to create, retrieve, update, and delete threaded comments and their replies, as well as manage their resolved status.

Managing Threaded Comments Threaded comments provide a series APIs. You can create, manage, and organize threaded comments and replies programmatically. Creating a Threaded Comment To create a threaded comment on a cell, use the add method on the sheet's threadedComments manager. Then add the first reply to the thread: Getting and Removing Threaded Comments You can get a threaded comment using the get method and remove it using the remove method: Managing Resolved Status You can mark a threaded comment as resolved or unresolved using the resolved method: Adding Replies Add replies to a threaded comment using the add method. Each reply can contain text, mentions, and links: Updating a Reply Update an existing reply using the set method: Removing a Reply Remove a specific reply using the remove method: User Management Configuring User Manager Before using threaded comments, configure the UserManager with user lookup and search functions: Setting Current User Set the current user who will author new comments and replies:
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 charset="utf-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <link rel="stylesheet" type="text/css" href="$DEMOROOT$/en/purejs/node_modules/@mescius/spread-sheets/styles/gc.spread.sheets.excel2013white.css"> <script src="$DEMOROOT$/en/purejs/node_modules/@mescius/spread-sheets/dist/gc.spread.sheets.all.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">Select a cell with threaded comment: <span id="commentTip" class="tc-comment-tip">No Threaded Comment</span> </h4> </div> <!-- Current User Selection --> <div class="tc-section"> <h5 class="tc-section-title">Current User</h5> <div class="tc-option-row"> <label class="tc-label">Switch User:</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> <!-- Thread Information --> <div class="tc-section"> <h5 class="tc-section-title">Thread Information</h5> <div class="tc-info-row"> <label class="tc-label">Location:</label> <span id="lblThreadLocation" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">Resolved:</label> <span id="lblThreadResolved" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">Replies Count:</label> <span id="lblThreadRepliesCount" class="tc-info-value">-</span> </div> </div> <!-- Thread Operations --> <div class="tc-section"> <h5 class="tc-section-title">Thread Operations</h5> <div class="tc-option-row"> <input type="text" id="txtNewThread" class="tc-input" placeholder="Enter thread message..." style="width: 100%; margin-bottom: 6px;" /> <button id="btnAddThread" class="tc-btn tc-btn-primary">Add New Thread</button> </div> <div class="tc-option-row"> <button id="btnToggleResolved" class="tc-btn tc-btn-secondary">Toggle Resolved Status</button> <button id="btnDeleteThread" class="tc-btn tc-btn-danger">Delete Thread</button> </div> </div> <!-- Replies Management --> <div class="tc-section"> <h5 class="tc-section-title">Replies Management</h5> <div class="tc-option-row"> <label class="tc-label">Select Reply:</label> <select id="repliesList" class="tc-select" size="4" style="width: 100%; margin-top: 4px;"> <option>No replies loaded</option> </select> </div> </div> <!-- Reply Information --> <div class="tc-section"> <h5 class="tc-section-title">Selected Reply Info</h5> <div class="tc-info-row"> <label class="tc-label">Author:</label> <span id="lblReplyAuthor" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">Created At:</label> <span id="lblReplyCreatedAt" class="tc-info-value">-</span> </div> <div class="tc-info-row"> <label class="tc-label">Message:</label> <span id="lblReplyMessage" class="tc-info-value">-</span> </div> </div> <!-- Reply Operations --> <div class="tc-section"> <h5 class="tc-section-title">Reply Operations</h5> <div class="tc-option-row"> <input type="text" id="txtNewReply" class="tc-input" placeholder="Enter reply message..." style="width: 100%; margin-bottom: 6px;" /> <button id="btnAddReply" class="tc-btn tc-btn-primary">Add Reply</button> </div> <div class="tc-option-row"> <input type="text" id="txtUpdateMessage" class="tc-input" placeholder="Enter new message..." style="width: 100%; margin-bottom: 6px;" /> <button id="btnUpdateReply" class="tc-btn tc-btn-secondary">Update Selected Reply</button> </div> <div class="tc-option-row"> <button id="btnDeleteReply" class="tc-btn tc-btn-danger">Delete Selected Reply</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; }