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;
}