jsdom
Version:
A JavaScript implementation of many web standards
119 lines (91 loc) • 3.54 kB
JavaScript
"use strict";
const DOMException = require("../generated/DOMException");
const { mixin } = require("../../utils");
const NodeImpl = require("./Node-impl").implementation;
const ChildNodeImpl = require("./ChildNode-impl").implementation;
const NonDocumentTypeChildNodeImpl = require("./NonDocumentTypeChildNode-impl").implementation;
const { TEXT_NODE } = require("../node-type");
const { MUTATION_TYPE, queueMutationRecord } = require("../helpers/mutation-observers");
// https://dom.spec.whatwg.org/#characterdata
class CharacterDataImpl extends NodeImpl {
constructor(globalObject, args, privateData) {
super(globalObject, args, privateData);
this._data = privateData.data;
}
// https://dom.spec.whatwg.org/#dom-characterdata-data
get data() {
return this._data;
}
set data(data) {
this.replaceData(0, this.length, data);
}
// https://dom.spec.whatwg.org/#dom-characterdata-length
get length() {
return this._data.length;
}
// https://dom.spec.whatwg.org/#dom-characterdata-substringdata
// https://dom.spec.whatwg.org/#concept-cd-substring
substringData(offset, count) {
const { length } = this;
if (offset > length) {
throw DOMException.create(this._globalObject, ["The index is not in the allowed range.", "IndexSizeError"]);
}
if (offset + count > length) {
return this._data.slice(offset);
}
return this._data.slice(offset, offset + count);
}
// https://dom.spec.whatwg.org/#dom-characterdata-appenddata
appendData(data) {
this.replaceData(this.length, 0, data);
}
// https://dom.spec.whatwg.org/#dom-characterdata-insertdata
insertData(offset, data) {
this.replaceData(offset, 0, data);
}
// https://dom.spec.whatwg.org/#dom-characterdata-deletedata
deleteData(offset, count) {
this.replaceData(offset, count, "");
}
// https://dom.spec.whatwg.org/#dom-characterdata-replacedata
// https://dom.spec.whatwg.org/#concept-cd-replace
replaceData(offset, count, data) {
const { length } = this;
if (offset > length) {
throw DOMException.create(this._globalObject, [
"The index is not in the allowed range.",
"IndexSizeError"
]);
}
if (offset + count > length) {
count = length - offset;
}
queueMutationRecord(MUTATION_TYPE.CHARACTER_DATA, this, null, null, this._data, [], [], null, null);
const start = this._data.slice(0, offset);
const end = this._data.slice(offset + count);
this._data = start + data + end;
for (const range of this._referencedRanges) {
const { _start, _end } = range;
if (_start.node === this && _start.offset > offset && _start.offset <= offset + count) {
range._setLiveRangeStart(this, offset);
}
if (_end.node === this && _end.offset > offset && _end.offset <= offset + count) {
range._setLiveRangeEnd(this, offset);
}
if (_start.node === this && _start.offset > offset + count) {
range._setLiveRangeStart(this, _start.offset + data.length - count);
}
if (_end.node === this && _end.offset > offset + count) {
range._setLiveRangeEnd(this, _end.offset + data.length - count);
}
}
if (this.nodeType === TEXT_NODE && this.parentNode) {
this.parentNode._childTextContentChangeSteps();
}
}
}
mixin(CharacterDataImpl.prototype, NonDocumentTypeChildNodeImpl.prototype);
mixin(CharacterDataImpl.prototype, ChildNodeImpl.prototype);
module.exports = {
implementation: CharacterDataImpl
};