// @ts-check const { RetainOp, InsertOp, RemoveOp } = require('./operation/scan_op') const Range = require('./range') /** * @import { CommentRawData } from "./types" * @import TextOperation from "./operation/text_operation" */ class Comment { /** * @readonly * @type {ReadonlyArray} */ ranges = [] /** * @readonly * @type {boolean} */ resolved = false /** * @param {string} id * @param {ReadonlyArray} ranges * @param {boolean} [resolved] */ constructor(id, ranges, resolved = false) { this.id = id this.resolved = resolved this.ranges = this.mergeRanges(ranges) } /** * * @param {number} cursor * @param {number} length * @param {boolean} [extendComment] * @returns {Comment} */ applyInsert(cursor, length, extendComment = false) { let existingRangeExtended = false const newRanges = [] for (const commentRange of this.ranges) { if (cursor === commentRange.end) { // insert right after the comment if (extendComment) { newRanges.push(commentRange.extendBy(length)) existingRangeExtended = true } else { newRanges.push(commentRange) } } else if (cursor === commentRange.start) { // insert at the start of the comment if (extendComment) { newRanges.push(commentRange.extendBy(length)) existingRangeExtended = true } else { newRanges.push(commentRange.moveBy(length)) } } else if (commentRange.startIsAfter(cursor)) { // insert before the comment newRanges.push(commentRange.moveBy(length)) } else if (commentRange.containsCursor(cursor)) { // insert is inside the comment if (extendComment) { newRanges.push(commentRange.extendBy(length)) existingRangeExtended = true } else { const [rangeUpToCursor, , rangeAfterCursor] = commentRange.insertAt( cursor, length ) // use current commentRange for the part before the cursor newRanges.push(new Range(commentRange.pos, rangeUpToCursor.length)) // add the part after the cursor as a new range newRanges.push(rangeAfterCursor) } } else { // insert is after the comment newRanges.push(commentRange) } } // if the insert is not inside any range, add a new range if (extendComment && !existingRangeExtended) { newRanges.push(new Range(cursor, length)) } return new Comment(this.id, newRanges, this.resolved) } /** * * @param {Range} deletedRange * @returns {Comment} */ applyDelete(deletedRange) { const newRanges = [] for (const commentRange of this.ranges) { if (commentRange.overlaps(deletedRange)) { newRanges.push(commentRange.subtract(deletedRange)) } else if (commentRange.startsAfter(deletedRange)) { newRanges.push(commentRange.moveBy(-deletedRange.length)) } else { newRanges.push(commentRange) } } return new Comment(this.id, newRanges, this.resolved) } /** * * @param {TextOperation} operation * @param {string} commentId * @returns {Comment} */ applyTextOperation(operation, commentId) { /** @type {Comment} */ let comment = this let cursor = 0 for (const op of operation.ops) { if (op instanceof RetainOp) { cursor += op.length } else if (op instanceof InsertOp) { comment = comment.applyInsert( cursor, op.insertion.length, op.commentIds?.includes(commentId) ) cursor += op.insertion.length } else if (op instanceof RemoveOp) { comment = comment.applyDelete(new Range(cursor, op.length)) } } return comment } isEmpty() { return this.ranges.length === 0 } /** * * @returns {CommentRawData} */ toRaw() { /** @type CommentRawData */ const raw = { id: this.id, ranges: this.ranges.map(range => range.toRaw()), } if (this.resolved) { raw.resolved = true } return raw } /** * @param {ReadonlyArray} ranges * @returns {ReadonlyArray} */ mergeRanges(ranges) { /** @type {Range[]} */ const mergedRanges = [] const sortedRanges = [...ranges].sort((a, b) => a.start - b.start) for (const range of sortedRanges) { if (range.isEmpty()) { continue } const lastMerged = mergedRanges[mergedRanges.length - 1] if (lastMerged?.overlaps(range)) { throw new Error('Ranges cannot overlap') } if (range.isEmpty()) { throw new Error('Comment range cannot be empty') } if (lastMerged?.canMerge(range)) { mergedRanges[mergedRanges.length - 1] = lastMerged.merge(range) } else { mergedRanges.push(range) } } return mergedRanges } /** * @param {CommentRawData} rawComment * @returns {Comment} */ static fromRaw(rawComment) { return new Comment( rawComment.id, rawComment.ranges.map(range => Range.fromRaw(range)), rawComment.resolved ) } } module.exports = Comment