Events

In Spread, when a Threaded Comment is changed, an event is triggered to notify about the data change. You can handle these changes by binding to the event and responding accordingly.

The ThreadedCommentChanging event occurs before the threaded comment changes (such as before replies are added/edited/removed, or before solved status changes). You can set args.cancel = true to prevent the change: The ThreadedCommentChanged event occurs when the threaded comment has changed (such as replies added/edited/removed, or solved status changed). The CellChanged event can also be used to detect when a threaded comment is added or removed from a cell: The UserMentioned event occurs when a reply containing one or more mentions is added to a threaded comment:
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}" }; function initSpread(spread) { // Configure UserManager first configureUserManager(); spread.suspendPaint(); var spreadNS = GC.Spread.Sheets; var sheet = spread.getActiveSheet(); sheet.options.allowCellOverflow = true; // Create a sample data table createSampleTable(sheet); // Add threaded comments to demonstrate various APIs addThreadedComments(sheet); // Add event listeners var threadedCommentEvent = document.getElementById("threadedCommentEvent"); // Listen for CellChanged event to detect threaded comment add/remove sheet.bind(spreadNS.Events.CellChanged, function(e, args) { if (args.propertyName === "threadedComment") { var text = threadedCommentEvent.value; if (text !== "") { text += '\n'; } var eventType; var hasOldValue = args.oldValue !== null && args.oldValue !== undefined; var hasNewValue = args.newValue !== null && args.newValue !== undefined; if (!hasOldValue && hasNewValue) { eventType = "Add threaded comment"; } else if (hasOldValue && !hasNewValue) { eventType = "Remove threaded comment"; } else if (hasOldValue && hasNewValue) { eventType = "Set threaded comment"; } else { eventType = "Threaded comment changed"; } threadedCommentEvent.value = text + "CellChanged: " + eventType + " at (" + args.row + ", " + args.col + ")"; } }); // Listen for ThreadedCommentChanged event with enhanced output sheet.bind(spreadNS.Events.ThreadedCommentChanged, function(e, args) { var text = threadedCommentEvent.value; if (text !== "") { text += '\n'; } var propertyName = args.propertyName; var eventDetail = "ThreadedCommentChanged at (" + args.threadedComment.row() + ", " + args.threadedComment.col() + "): " + propertyName; // Enhance output based on property name and value if (propertyName === "resolved") { eventDetail += " = " + (args.newValue ? "true" : "false"); } else if (propertyName === "reply") { var hasOldValue = args.oldValue !== null && args.oldValue !== undefined; var hasNewValue = args.newValue !== null && args.newValue !== undefined; // Get reply index information var replyIndex = args.replyIndex; if (!hasOldValue && hasNewValue) { eventDetail += " (Add reply at index " + replyIndex + ")"; } else if (hasOldValue && !hasNewValue) { eventDetail += " (Remove reply at index " + replyIndex + ")"; } else if (hasOldValue && hasNewValue) { eventDetail += " (Edit reply at index " + replyIndex + ")"; } } threadedCommentEvent.value = text + eventDetail; }); // Listen for UserMentioned event with enhanced output sheet.bind(spreadNS.Events.UserMentioned, function(e, args) { var text = threadedCommentEvent.value; if (text !== "") { text += '\n'; } var context = args.context; var userIds = args.userIds; // Create an array of promises to get all user information var userPromises = []; for (var i = 0; i < userIds.length; i++) { userPromises.push(GC.Spread.Common.UserManager.get(userIds[i])); } // Wait for all promises to resolve Promise.all(userPromises).then(function(users) { var userNames = []; for (var j = 0; j < users.length; j++) { var user = users[j]; userNames.push((user && user.name) || "Can't find the user who's id is " + userIds[j]); } var pluralText = userNames.length > 1 ? "are" : "is"; var eventDetail = userNames.join(", ") + " " + pluralText + " mentioned at (" + context.threadedComment.row() + ", " + context.threadedComment.col() + ")"; threadedCommentEvent.value = text + eventDetail; }); }); spread.resumePaint(); } 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: "" } } ]; // Configure UserManager with mock implementation GC.Spread.Common.UserManager.configure({ get: async function(userId) { if (userId === undefined) { return; } return new Promise(function(resolve) { var user = users.find(function(u) { return u.id === userId; }); resolve(user); }); }, search: async function(query) { return new Promise(function(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); // Display current user in B1 sheet.setValue(0, 1, "Current User: Alice Johnson"); sheet.getCell(0, 1).font("bold 11pt Arial").foreColor("#4A90D9"); // 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"); }
<!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="sample-tutorial"> <div id="ss" class="sample-spreadsheets"></div> <div class="options-container"> <textarea id="threadedCommentEvent" style="width: 100%; height: 100%" readOnly placeholder="Event log: Threaded Comment events will be displayed here..."></textarea> </div> </div> </body> </html>
.sample-tutorial { position: relative; height: 100%; overflow: hidden; } .sample-spreadsheets { width: calc(100% - 280px); height: 100%; overflow: hidden; float: left; } .options-container { float: right; width: 280px; overflow: auto; padding: 12px; height: 100%; box-sizing: border-box; background: #fbfbfb; } textarea { width: 100%; height: 80px; padding: 6px 12px; box-sizing: border-box; } .sample-options { z-index: 1000; } body { position: absolute; top: 0; bottom: 0; left: 0; right: 0; }