summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorCashew <52880648+Scrub1492@users.noreply.github.com>2024-01-01 08:08:43 +0700
committerGitHub <noreply@github.com>2024-01-01 01:08:43 +0000
commitb5f2a36597da9920081c65f826404cf054627745 (patch)
treedc6ac016ce72710c63e5f39ede0cff0459362802
parent7303da3991814a0ce220bf2fff3e51b968913b86 (diff)
document-util to use JSON attribute instead of multiple data-* attributes (#492)
* data-* to json * update tests * fix tests * fix tests * test fixes * fix types * fix types
-rw-r--r--test/data/html/document-util.html446
-rw-r--r--test/document-util.test.js39
-rw-r--r--types/test/document-util.d.ts42
3 files changed, 305 insertions, 222 deletions
diff --git a/test/data/html/document-util.html b/test/data/html/document-util.html
index d8ac012e..8e007392 100644
--- a/test/data/html/document-util.html
+++ b/test/data/html/document-util.html
@@ -15,183 +15,207 @@
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="0"
- data-end-node-selector="span"
- data-end-offset="0"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="真白「心配してくださって、ありがとございます」"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 0,
+ "endNodeSelector": "span",
+ "endOffset": 0,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "真白「心配してくださって、ありがとございます」"
+ }'
>
<span>真白「心配してくださって、ありがとございます」</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="5"
- data-end-node-selector="span"
- data-end-offset="5"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="心配してくださって、ありがとございます"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 5,
+ "endNodeSelector": "span",
+ "endOffset": 5,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "心配してくださって、ありがとございます"
+ }'
>
<span>真白「心配してくださって、ありがとございます」</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="16"
- data-end-node-selector="span"
- data-end-offset="16"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="心配して「くださって」、ありがと「ございます」"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 16,
+ "endNodeSelector": "span",
+ "endOffset": 16,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "心配して「くださって」、ありがと「ございます」"
+ }'
>
<span>真白「心配して「くださって」、ありがと「ございます」」</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="4"
- data-end-node-selector="span"
- data-end-offset="4"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="ありがとございます。"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 4,
+ "endNodeSelector": "span",
+ "endOffset": 4,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "ありがとございます。"
+ }'
>
<span>ありがとございます。ありがとございます。</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="14"
- data-end-node-selector="span"
- data-end-offset="14"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="ありがとございます。"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 14,
+ "endNodeSelector": "span",
+ "endOffset": 14,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "ありがとございます。"
+ }'
>
<span>ありがとございます。ありがとございます。</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="4"
- data-end-node-selector="span"
- data-end-offset="4"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="ありがとございます。!?"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 4,
+ "endNodeSelector": "span",
+ "endOffset": 4,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "ありがとございます。!?"
+ }'
>
<span>ありがとございます。!?ありがとございます。!?</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span"
- data-caret-range-from-point-selector="span"
- data-start-node-selector="span"
- data-start-offset="4"
- data-end-node-selector="span"
- data-end-offset="4"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="ありがとございます!!!"
+ data-test-data='{
+ "elementFromPointSelector": "span",
+ "caretRangeFromPointSelector": "span",
+ "startNodeSelector": "span",
+ "startOffset": 4,
+ "endNodeSelector": "span",
+ "endOffset": 4,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "ありがとございます!!!"
+ }'
>
<span>ありがとございます!!!ありがとございます!!!</span>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="input"
- data-caret-range-from-point-selector="input"
- data-start-node-selector="input"
- data-start-offset="0"
- data-end-node-selector="input"
- data-end-offset="0"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="真白「心配してくださって、ありがとございます」"
- data-has-imposter="true"
+ data-test-data='{
+ "elementFromPointSelector": "input",
+ "caretRangeFromPointSelector": "input",
+ "startNodeSelector": "input",
+ "startOffset": 0,
+ "endNodeSelector": "input",
+ "endOffset": 0,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "真白「心配してくださって、ありがとございます」",
+ "hasImposter": true
+ }'
>
<input type="text" value="真白「心配してくださって、ありがとございます」" style="width: 100%; box-sizing: border-box; font-family: inherit; font-size: inherit; border: 1px solid #d8d8d8; padding: 0.2em;">
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="textarea"
- data-caret-range-from-point-selector="textarea"
- data-start-node-selector="textarea"
- data-start-offset="0"
- data-end-node-selector="textarea"
- data-end-offset="0"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="100"
- data-sentence="真白「心配してくださって、ありがとございます」"
- data-has-imposter="true"
+ data-test-data='{
+ "elementFromPointSelector": "textarea",
+ "caretRangeFromPointSelector": "textarea",
+ "startNodeSelector": "textarea",
+ "startOffset": 0,
+ "endNodeSelector": "textarea",
+ "endOffset": 0,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 100,
+ "sentence": "真白「心配してくださって、ありがとございます」",
+ "hasImposter": true
+ }'
>
<textarea style="width: 100%; height: 3em; box-sizing: border-box; font-family: inherit; font-size: inherit; border: 1px solid #d8d8d8; padding: 0.2em;">真白「心配してくださって、ありがとございます」</textarea>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="button"
- data-caret-range-from-point-selector="button"
- data-start-node-selector="button"
- data-start-offset="0"
- data-end-node-selector="button"
- data-end-offset="0"
- data-result-type="TextSourceElement"
- data-sentence-scan-extent="100"
- data-sentence="よみたん"
+ data-test-data='{
+ "elementFromPointSelector": "button",
+ "caretRangeFromPointSelector": "button",
+ "startNodeSelector": "button",
+ "startOffset": 0,
+ "endNodeSelector": "button",
+ "endOffset": 0,
+ "resultType": "TextSourceElement",
+ "sentenceScanExtent": 100,
+ "sentence": "よみたん"
+ }'
>
<button type="button" style="width: 100%; box-sizing: border-box; font-family: inherit; font-size: inherit; border: 1px solid #d8d8d8; background-color: #f0f0f0; padding: 0.2em;">よみたん</button>
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="img"
- data-caret-range-from-point-selector="img"
- data-start-node-selector="img"
- data-start-offset="0"
- data-end-node-selector="img"
- data-end-offset="0"
- data-result-type="TextSourceElement"
- data-sentence-scan-extent="100"
- data-sentence="よみたん"
+ data-test-data='{
+ "elementFromPointSelector": "img",
+ "caretRangeFromPointSelector": "img",
+ "startNodeSelector": "img",
+ "startOffset": 0,
+ "endNodeSelector": "img",
+ "endOffset": 0,
+ "resultType": "TextSourceElement",
+ "sentenceScanExtent": 100,
+ "sentence": "よみたん"
+ }'
>
<img src="" alt="よみたん" title="よみたん" style="width: 70px; height: 70px; image-rendering: crisp-edges; image-rendering: pixelated; display: block;">
</test-case>
<test-case
data-test-type="scan"
- data-element-from-point-selector="span:nth-of-type(3)"
- data-caret-range-from-point-selector="span:nth-of-type(3)"
- data-start-node-selector="span:nth-of-type(3)"
- data-start-offset="0"
- data-end-node-selector="span:nth-of-type(3)"
- data-end-offset="0"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="22"
- data-sentence="ありがとございます3"
- data-terminate-at-newlines="true"
+ data-test-data='{
+ "elementFromPointSelector": "span:nth-of-type(3)",
+ "caretRangeFromPointSelector": "span:nth-of-type(3)",
+ "startNodeSelector": "span:nth-of-type(3)",
+ "startOffset": 0,
+ "endNodeSelector": "span:nth-of-type(3)",
+ "endOffset": 0,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 22,
+ "sentence": "ありがとございます3",
+ "terminateAtNewlines": true
+ }'
>
<span>ありがとございます1</span>
<span>ありがとございます2</span>
@@ -202,16 +226,18 @@
<test-case
data-test-type="scan"
- data-element-from-point-selector="span:nth-of-type(3)"
- data-caret-range-from-point-selector="span:nth-of-type(3)"
- data-start-node-selector="span:nth-of-type(3)"
- data-start-offset="0"
- data-end-node-selector="span:nth-of-type(3)"
- data-end-offset="0"
- data-result-type="TextSourceRange"
- data-sentence-scan-extent="22"
- data-sentence="ありがとございます1&#10;ありがとございます2&#10;ありがとございます3&#10;ありがとございます4"
- data-terminate-at-newlines="false"
+ data-test-data='{
+ "elementFromPointSelector": "span:nth-of-type(3)",
+ "caretRangeFromPointSelector": "span:nth-of-type(3)",
+ "startNodeSelector": "span:nth-of-type(3)",
+ "startOffset": 0,
+ "endNodeSelector": "span:nth-of-type(3)",
+ "endOffset": 0,
+ "resultType": "TextSourceRange",
+ "sentenceScanExtent": 22,
+ "sentence": "ありがとございます1\nありがとございます2\nありがとございます3\nありがとございます4",
+ "terminateAtNewlines": false
+ }'
>
<span>ありがとございます1</span>
<span>ありがとございます2</span>
@@ -221,27 +247,31 @@
</test-case>
<test-case
- data-test-type="text-source-range-seek"
- data-seek-node-selector="span:nth-of-type(1)"
- data-seek-node-is-text="true"
- data-seek-offset="0"
- data-seek-length="149"
- data-seek-direction="forward"
- data-expected-result-node-selector="span:nth-of-type(1)"
- data-expected-result-node-is-text="true"
- data-expected-result-offset="149"
- data-expected-result-content="
- あいうえお
- かきくけこ
- さしすせそ
- たちつてと
- なにぬねの
- はひふへほ
- まみむめも
- や&#x3000;ゆ&#x3000;よ
- らりるれろ
- わゐ&#x3000;ゑを
- "
+ data-test-type="text-source-range-seek"
+ data-test-data='{
+ "seekNodeSelector": "span:nth-of-type(1)",
+ "seekNodeIsText": true,
+ "seekOffset": 0,
+ "seekLength": 149,
+ "seekDirection": "forward",
+ "expectedResultNodeSelector": "span:nth-of-type(1)",
+ "expectedResultNodeIsText": true,
+ "expectedResultOffset": 149,
+ "expectedResultContent": [
+ "",
+ " あいうえお",
+ " かきくけこ",
+ " さしすせそ",
+ " たちつてと",
+ " なにぬねの",
+ " はひふへほ",
+ " まみむめも",
+ " や ゆ よ",
+ " らりるれろ",
+ " わゐ ゑを",
+ " "
+ ]
+ }'
>
<span>
あいうえお
@@ -259,26 +289,30 @@
<test-case
data-test-type="text-source-range-seek"
- data-seek-node-selector="span:nth-of-type(1)"
- data-seek-node-is-text="true"
- data-seek-offset="149"
- data-seek-length="149"
- data-seek-direction="backward"
- data-expected-result-node-selector="span:nth-of-type(1)"
- data-expected-result-node-is-text="true"
- data-expected-result-offset="0"
- data-expected-result-content="
- あいうえお
- かきくけこ
- さしすせそ
- たちつてと
- なにぬねの
- はひふへほ
- まみむめも
- や&#x3000;ゆ&#x3000;よ
- らりるれろ
- わゐ&#x3000;ゑを
- "
+ data-test-data='{
+ "seekNodeSelector": "span:nth-of-type(1)",
+ "seekNodeIsText": true,
+ "seekOffset": 149,
+ "seekLength": 149,
+ "seekDirection": "backward",
+ "expectedResultNodeSelector": "span:nth-of-type(1)",
+ "expectedResultNodeIsText": true,
+ "expectedResultOffset": 0,
+ "expectedResultContent": [
+ "",
+ " あいうえお",
+ " かきくけこ",
+ " さしすせそ",
+ " たちつてと",
+ " なにぬねの",
+ " はひふへほ",
+ " まみむめも",
+ " や ゆ よ",
+ " らりるれろ",
+ " わゐ ゑを",
+ " "
+ ]
+ }'
>
<span>
あいうえお
@@ -296,26 +330,30 @@
<test-case
data-test-type="text-source-range-seek"
- data-seek-node-selector="span:nth-of-type(1)"
- data-seek-node-is-text="true"
- data-seek-offset="0"
- data-seek-length="150"
- data-seek-direction="forward"
- data-expected-result-node-selector="span:nth-of-type(2)"
- data-expected-result-node-is-text="true"
- data-expected-result-offset="1"
- data-expected-result-content="
- あいうえお
- かきくけこ
- さしすせそ
- たちつてと
- なにぬねの
- はひふへほ
- まみむめも
- や&#x3000;ゆ&#x3000;よ
- らりるれろ
- わゐ&#x3000;ゑを
- t"
+ data-test-data='{
+ "seekNodeSelector": "span:nth-of-type(1)",
+ "seekNodeIsText": true,
+ "seekOffset": 0,
+ "seekLength": 150,
+ "seekDirection": "forward",
+ "expectedResultNodeSelector": "span:nth-of-type(2)",
+ "expectedResultNodeIsText": true,
+ "expectedResultOffset": 1,
+ "expectedResultContent": [
+ "",
+ " あいうえお",
+ " かきくけこ",
+ " さしすせそ",
+ " たちつてと",
+ " なにぬねの",
+ " はひふへほ",
+ " まみむめも",
+ " や ゆ よ",
+ " らりるれろ",
+ " わゐ ゑを",
+ " t"
+ ]
+ }'
>
<span>
あいうえお
@@ -333,26 +371,30 @@
<test-case
data-test-type="text-source-range-seek"
- data-seek-node-selector="span:nth-of-type(2)"
- data-seek-node-is-text="true"
- data-seek-offset="1"
- data-seek-length="150"
- data-seek-direction="backward"
- data-expected-result-node-selector="span:nth-of-type(1)"
- data-expected-result-node-is-text="true"
- data-expected-result-offset="0"
- data-expected-result-content="
- あいうえお
- かきくけこ
- さしすせそ
- たちつてと
- なにぬねの
- はひふへほ
- まみむめも
- や&#x3000;ゆ&#x3000;よ
- らりるれろ
- わゐ&#x3000;ゑを
- t"
+ data-test-data='{
+ "seekNodeSelector": "span:nth-of-type(2)",
+ "seekNodeIsText": true,
+ "seekOffset": 1,
+ "seekLength": 150,
+ "seekDirection": "bacward",
+ "expectedResultNodeSelector": "span:nth-of-type(1)",
+ "expectedResultNodeIsText": true,
+ "expectedResultOffset": 0,
+ "expectedResultContent": [
+ "",
+ " あいうえお",
+ " かきくけこ",
+ " さしすせそ",
+ " たちつてと",
+ " なにぬねの",
+ " はひふへほ",
+ " まみむめも",
+ " や ゆ よ",
+ " らりるれろ",
+ " わゐ ゑを",
+ " t"
+ ]
+ }'
>
<span>
あいうえお
diff --git a/test/document-util.test.js b/test/document-util.test.js
index 109345d1..cc8db706 100644
--- a/test/document-util.test.js
+++ b/test/document-util.test.js
@@ -24,6 +24,7 @@ import {DOMTextScanner} from '../ext/js/dom/dom-text-scanner.js';
import {TextSourceElement} from '../ext/js/dom/text-source-element.js';
import {TextSourceRange} from '../ext/js/dom/text-source-range.js';
import {createDomTest} from './fixtures/dom-test.js';
+import {parseJson} from '../dev/json.js';
const dirname = path.dirname(fileURLToPath(import.meta.url));
@@ -116,6 +117,7 @@ describe('DocumentUtil', () => {
const {document} = window;
for (const testElement of /** @type {NodeListOf<HTMLElement>} */ (document.querySelectorAll('test-case[data-test-type=scan]'))) {
// Get test parameters
+ /** @type {import('test/document-util').DocumentUtilTestData} */
const {
elementFromPointSelector,
caretRangeFromPointSelector,
@@ -128,17 +130,15 @@ describe('DocumentUtil', () => {
sentence,
hasImposter,
terminateAtNewlines
- } = testElement.dataset;
+ } = parseJson(/** @type {string} */ (testElement.dataset.testData));
const elementFromPointValue = querySelectorChildOrSelf(testElement, elementFromPointSelector);
const caretRangeFromPointValue = querySelectorChildOrSelf(testElement, caretRangeFromPointSelector);
const startNode = getChildTextNodeOrSelf(window, querySelectorChildOrSelf(testElement, startNodeSelector));
const endNode = getChildTextNodeOrSelf(window, querySelectorChildOrSelf(testElement, endNodeSelector));
- const startOffset2 = parseInt(/** @type {string} */ (startOffset), 10);
- const endOffset2 = parseInt(/** @type {string} */ (endOffset), 10);
- const sentenceScanExtent2 = parseInt(/** @type {string} */ (sentenceScanExtent), 10);
- const terminateAtNewlines2 = (terminateAtNewlines !== 'false');
+ // Defaults to true
+ const terminateAtNewlines2 = typeof terminateAtNewlines === 'boolean' ? terminateAtNewlines : true;
expect(elementFromPointValue).not.toStrictEqual(null);
expect(caretRangeFromPointValue).not.toStrictEqual(null);
@@ -150,11 +150,11 @@ describe('DocumentUtil', () => {
document.caretRangeFromPoint = (x, y) => {
const imposter = getChildTextNodeOrSelf(window, findImposterElement(document));
- expect(!!imposter).toStrictEqual(hasImposter === 'true');
+ expect(!!imposter).toStrictEqual(!!hasImposter);
const range = document.createRange();
- range.setStart(/** @type {Node} */ (imposter ? imposter : startNode), startOffset2);
- range.setEnd(/** @type {Node} */ (imposter ? imposter : startNode), endOffset2);
+ range.setStart(/** @type {Node} */ (imposter ? imposter : startNode), startOffset);
+ range.setEnd(/** @type {Node} */ (imposter ? imposter : startNode), endOffset);
// Override getClientRects to return a rect guaranteed to contain (x, y)
range.getClientRects = () => {
@@ -214,7 +214,7 @@ describe('DocumentUtil', () => {
const sentenceActual = DocumentUtil.extractSentence(
source,
false,
- sentenceScanExtent2,
+ sentenceScanExtent,
terminateAtNewlines2,
terminatorMap,
forwardQuoteMap,
@@ -233,6 +233,7 @@ describe('DOMTextScanner', () => {
const {document} = window;
for (const testElement of /** @type {NodeListOf<HTMLElement>} */ (document.querySelectorAll('test-case[data-test-type=text-source-range-seek]'))) {
// Get test parameters
+ /** @type {import('test/document-util').DOMTextScannerTestData} */
const {
seekNodeSelector,
seekNodeIsText,
@@ -243,33 +244,31 @@ describe('DOMTextScanner', () => {
expectedResultNodeIsText,
expectedResultOffset,
expectedResultContent
- } = testElement.dataset;
-
- const seekOffset2 = parseInt(/** @type {string} */ (seekOffset), 10);
- const seekLength2 = parseInt(/** @type {string} */ (seekLength), 10);
- const expectedResultOffset2 = parseInt(/** @type {string} */ (expectedResultOffset), 10);
+ } = parseJson(/** @type {string} */ (testElement.dataset.testData));
/** @type {?Node} */
let seekNode = testElement.querySelector(/** @type {string} */ (seekNodeSelector));
- if (seekNodeIsText === 'true' && seekNode !== null) {
+ if (seekNodeIsText && seekNode !== null) {
seekNode = seekNode.firstChild;
}
+ const expectedResultContent2 = expectedResultContent.join('\n');
+
/** @type {?Node} */
let expectedResultNode = testElement.querySelector(/** @type {string} */ (expectedResultNodeSelector));
- if (expectedResultNodeIsText === 'true' && expectedResultNode !== null) {
+ if (expectedResultNodeIsText && expectedResultNode !== null) {
expectedResultNode = expectedResultNode.firstChild;
}
const {node, offset, content} = (
seekDirection === 'forward' ?
- new DOMTextScanner(/** @type {Node} */ (seekNode), seekOffset2, true, false).seek(seekLength2) :
- new DOMTextScanner(/** @type {Node} */ (seekNode), seekOffset2, true, false).seek(-seekLength2)
+ new DOMTextScanner(/** @type {Node} */ (seekNode), seekOffset, true, false).seek(seekLength) :
+ new DOMTextScanner(/** @type {Node} */ (seekNode), seekOffset, true, false).seek(-seekLength)
);
expect(node).toStrictEqual(expectedResultNode);
- expect(offset).toStrictEqual(expectedResultOffset2);
- expect(content).toStrictEqual(expectedResultContent);
+ expect(offset).toStrictEqual(expectedResultOffset);
+ expect(content).toStrictEqual(expectedResultContent2);
}
});
});
diff --git a/types/test/document-util.d.ts b/types/test/document-util.d.ts
new file mode 100644
index 00000000..3c09f7f0
--- /dev/null
+++ b/types/test/document-util.d.ts
@@ -0,0 +1,42 @@
+/*
+ * Copyright (C) 2023 Yomitan Authors
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+export type DocumentUtilTestData = {
+ elementFromPointSelector: string;
+ caretRangeFromPointSelector: string;
+ startNodeSelector: string;
+ startOffset: number;
+ endNodeSelector: string;
+ endOffset: number;
+ resultType: string;
+ sentenceScanExtent: number;
+ sentence: string;
+ hasImposter: boolean | undefined;
+ terminateAtNewlines: boolean | undefined;
+};
+
+export type DOMTextScannerTestData = {
+ seekNodeSelector: string;
+ seekNodeIsText: boolean;
+ seekOffset: number;
+ seekLength: number;
+ seekDirection: string;
+ expectedResultNodeSelector: string;
+ expectedResultNodeIsText: boolean;
+ expectedResultOffset: number;
+ expectedResultContent: string[];
+};