@hamster-note/document-parser 0.1.0 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,19 @@
1
+ import type { IntermediateDocument } from '@hamster-note/types';
2
+ export type ParserInput = ArrayBuffer | ArrayBufferView | Blob;
3
+ export declare abstract class DocumentParser {
4
+ static readonly exts: readonly string[];
5
+ /**
6
+ * 将原始输入解析为中间文档结构。
7
+ * - 输入必须是内存态数据(ArrayBuffer / ArrayBufferView / Blob)
8
+ * - 失败时抛出异常;不要返回 undefined。
9
+ */
10
+ abstract encode(input: ParserInput): Promise<IntermediateDocument>;
11
+ /**
12
+ * 可选:从中间文档逆序列化回原始文件数据。
13
+ * 默认不支持,需要的子类自行实现。
14
+ */
15
+ decode(_intermediateDocument: IntermediateDocument): Promise<ParserInput>;
16
+ protected static toArrayBuffer(input: ParserInput): Promise<ArrayBuffer>;
17
+ protected static toUint8Array(input: ParserInput): Promise<Uint8Array>;
18
+ }
19
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/Parser/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAA;AAG/D,MAAM,MAAM,WAAW,GAAG,WAAW,GAAG,eAAe,GAAG,IAAI,CAAA;AAE9D,8BAAsB,cAAc;IAElC,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,SAAS,MAAM,EAAE,CAAK;IAE5C;;;;OAIG;IACH,QAAQ,CAAC,MAAM,CAAC,KAAK,EAAE,WAAW,GAAG,OAAO,CAAC,oBAAoB,CAAC;IAElE;;;OAGG;IACH,MAAM,CAAC,qBAAqB,EAAE,oBAAoB,GAAG,OAAO,CAAC,WAAW,CAAC;qBAKlD,aAAa,CAClC,KAAK,EAAE,WAAW,GACjB,OAAO,CAAC,WAAW,CAAC;qBAmBA,YAAY,CAAC,KAAK,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,CAAC;CAI7E"}
@@ -0,0 +1,38 @@
1
+ export class DocumentParser {
2
+ /**
3
+ * 可选:从中间文档逆序列化回原始文件数据。
4
+ * 默认不支持,需要的子类自行实现。
5
+ */
6
+ decode(_intermediateDocument) {
7
+ return Promise.reject(new Error('decode is not implemented'));
8
+ }
9
+ // 将各种输入转换为 ArrayBuffer,方便子类直接消费二进制
10
+ static async toArrayBuffer(input) {
11
+ if (input instanceof ArrayBuffer) {
12
+ return input;
13
+ }
14
+ if (ArrayBuffer.isView(input)) {
15
+ // 拷贝为独立的 ArrayBuffer,避免 SharedArrayBuffer 兼容性问题
16
+ const buffer = new ArrayBuffer(input.byteLength);
17
+ new Uint8Array(buffer).set(new Uint8Array(input.buffer, input.byteOffset, input.byteLength));
18
+ return buffer;
19
+ }
20
+ if (typeof Blob !== 'undefined' && input instanceof Blob) {
21
+ return input.arrayBuffer();
22
+ }
23
+ throw new TypeError('Unsupported parser input');
24
+ }
25
+ // 子类若需要 Uint8Array,可直接复用
26
+ static async toUint8Array(input) {
27
+ const buffer = await DocumentParser.toArrayBuffer(input);
28
+ return new Uint8Array(buffer);
29
+ }
30
+ }
31
+ // 子类必须声明支持的扩展名列表(不带点),用于注册分发
32
+ Object.defineProperty(DocumentParser, "exts", {
33
+ enumerable: true,
34
+ configurable: true,
35
+ writable: true,
36
+ value: []
37
+ });
38
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/Parser/index.ts"],"names":[],"mappings":"AAKA,MAAM,OAAgB,cAAc;IAWlC;;;OAGG;IACH,MAAM,CAAC,qBAA2C;QAChD,OAAO,OAAO,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,2BAA2B,CAAC,CAAC,CAAA;IAC/D,CAAC;IAED,mCAAmC;IACzB,MAAM,CAAC,KAAK,CAAC,aAAa,CAClC,KAAkB;QAElB,IAAI,KAAK,YAAY,WAAW,EAAE,CAAC;YACjC,OAAO,KAAK,CAAA;QACd,CAAC;QACD,IAAI,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;YAC9B,gDAAgD;YAChD,MAAM,MAAM,GAAG,IAAI,WAAW,CAAC,KAAK,CAAC,UAAU,CAAC,CAAA;YAChD,IAAI,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,CACxB,IAAI,UAAU,CAAC,KAAK,CAAC,MAAM,EAAE,KAAK,CAAC,UAAU,EAAE,KAAK,CAAC,UAAU,CAAC,CACjE,CAAA;YACD,OAAO,MAAM,CAAA;QACf,CAAC;QACD,IAAI,OAAO,IAAI,KAAK,WAAW,IAAI,KAAK,YAAY,IAAI,EAAE,CAAC;YACzD,OAAO,KAAK,CAAC,WAAW,EAAE,CAAA;QAC5B,CAAC;QACD,MAAM,IAAI,SAAS,CAAC,0BAA0B,CAAC,CAAA;IACjD,CAAC;IAED,yBAAyB;IACf,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,KAAkB;QACpD,MAAM,MAAM,GAAG,MAAM,cAAc,CAAC,aAAa,CAAC,KAAK,CAAC,CAAA;QACxD,OAAO,IAAI,UAAU,CAAC,MAAM,CAAC,CAAA;IAC/B,CAAC;;AA3CD,6BAA6B;AACb;;;;WAA0B,EAAE;GAAA"}
@@ -0,0 +1,4 @@
1
+ export * from './types/DocumentAnchor';
2
+ export * from './Parser';
3
+ export * from './register';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,wBAAwB,CAAA;AACtC,cAAc,UAAU,CAAA;AACxB,cAAc,YAAY,CAAA"}
package/dist/index.js ADDED
@@ -0,0 +1,4 @@
1
+ export * from './types/DocumentAnchor';
2
+ export * from './Parser';
3
+ export * from './register';
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,wBAAwB,CAAA;AACtC,cAAc,UAAU,CAAA;AACxB,cAAc,YAAY,CAAA"}
@@ -0,0 +1,20 @@
1
+ import type { DocumentParser, ParserInput } from '../Parser';
2
+ import type { IntermediateDocument } from '@hamster-note/types';
3
+ export interface DocumentParserConstructor<T extends DocumentParser = DocumentParser> {
4
+ new (): T;
5
+ readonly exts: readonly string[];
6
+ }
7
+ export declare const registerParser: (Parser: DocumentParserConstructor) => void;
8
+ export declare function unregisterParser(Parser: DocumentParserConstructor): void;
9
+ export declare function getParserByExt(ext?: string): DocumentParserConstructor | undefined;
10
+ export interface ParseOptions {
11
+ filename?: string;
12
+ ext?: string;
13
+ parser?: DocumentParserConstructor;
14
+ }
15
+ export declare function parse(input: ParserInput, options?: ParseOptions): Promise<IntermediateDocument>;
16
+ export declare function listRegisteredParsers(): {
17
+ ext: string;
18
+ parser: DocumentParserConstructor<DocumentParser>;
19
+ }[];
20
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/register/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,WAAW,EAAE,MAAM,WAAW,CAAA;AAC5D,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAA;AAE/D,MAAM,WAAW,yBAAyB,CACxC,CAAC,SAAS,cAAc,GAAG,cAAc;IAEzC,QAAQ,CAAC,CAAA;IACT,QAAQ,CAAC,IAAI,EAAE,SAAS,MAAM,EAAE,CAAA;CACjC;AAOD,eAAO,MAAM,cAAc,GAAI,QAAQ,yBAAyB,SAY/D,CAAA;AAED,wBAAgB,gBAAgB,CAAC,MAAM,EAAE,yBAAyB,QAQjE;AAED,wBAAgB,cAAc,CAC5B,GAAG,CAAC,EAAE,MAAM,GACX,yBAAyB,GAAG,SAAS,CAGvC;AAED,MAAM,WAAW,YAAY;IAC3B,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,GAAG,CAAC,EAAE,MAAM,CAAA;IACZ,MAAM,CAAC,EAAE,yBAAyB,CAAA;CACnC;AAED,wBAAsB,KAAK,CACzB,KAAK,EAAE,WAAW,EAClB,OAAO,GAAE,YAAiB,GACzB,OAAO,CAAC,oBAAoB,CAAC,CAe/B;AAGD,wBAAgB,qBAAqB;;;IAEpC"}
@@ -0,0 +1,48 @@
1
+ const parserMap = new Map();
2
+ const normalizeExt = (ext) => ext.replace(/^\./, '').toLowerCase();
3
+ // 注册解析器:防止重复注册同扩展名的不同类
4
+ export const registerParser = (Parser) => {
5
+ if (!Array.isArray(Parser.exts) || Parser.exts.length === 0) {
6
+ throw new Error('Parser.exts 必须是非空数组');
7
+ }
8
+ Parser.exts.forEach((ext) => {
9
+ const key = normalizeExt(ext);
10
+ const exists = parserMap.get(key);
11
+ if (exists && exists !== Parser) {
12
+ throw new Error(`扩展名 ${key} 已被其他解析器占用`);
13
+ }
14
+ parserMap.set(key, Parser);
15
+ });
16
+ };
17
+ export function unregisterParser(Parser) {
18
+ Parser.exts.forEach((ext) => {
19
+ const key = normalizeExt(ext);
20
+ const exists = parserMap.get(key);
21
+ if (exists === Parser) {
22
+ parserMap.delete(key);
23
+ }
24
+ });
25
+ }
26
+ export function getParserByExt(ext) {
27
+ if (!ext)
28
+ return undefined;
29
+ return parserMap.get(normalizeExt(ext));
30
+ }
31
+ export async function parse(input, options = {}) {
32
+ var _a, _b, _c;
33
+ const extFromName = (_a = options.filename) === null || _a === void 0 ? void 0 : _a.split('.').pop();
34
+ const ext = (_b = options.ext) !== null && _b !== void 0 ? _b : extFromName;
35
+ const Parser = (_c = options.parser) !== null && _c !== void 0 ? _c : getParserByExt(ext);
36
+ if (!Parser) {
37
+ throw new Error(`未找到匹配的解析器,扩展名: ${ext !== null && ext !== void 0 ? ext : '未知'},已注册: ${[
38
+ ...parserMap.keys()
39
+ ].join(', ')}`);
40
+ }
41
+ const parser = new Parser();
42
+ return parser.encode(input);
43
+ }
44
+ // 便于测试或运行时可观测性
45
+ export function listRegisteredParsers() {
46
+ return [...parserMap.entries()].map(([ext, ctor]) => ({ ext, parser: ctor }));
47
+ }
48
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/register/index.ts"],"names":[],"mappings":"AAUA,MAAM,SAAS,GAA2C,IAAI,GAAG,EAAE,CAAA;AAEnE,MAAM,YAAY,GAAG,CAAC,GAAW,EAAE,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,WAAW,EAAE,CAAA;AAE1E,uBAAuB;AACvB,MAAM,CAAC,MAAM,cAAc,GAAG,CAAC,MAAiC,EAAE,EAAE;IAClE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC5D,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAA;IACxC,CAAC;IACD,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;QAC1B,MAAM,GAAG,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAC7B,MAAM,MAAM,GAAG,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;QACjC,IAAI,MAAM,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;YAChC,MAAM,IAAI,KAAK,CAAC,OAAO,GAAG,YAAY,CAAC,CAAA;QACzC,CAAC;QACD,SAAS,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,CAAA;IAC5B,CAAC,CAAC,CAAA;AACJ,CAAC,CAAA;AAED,MAAM,UAAU,gBAAgB,CAAC,MAAiC;IAChE,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;QAC1B,MAAM,GAAG,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAC7B,MAAM,MAAM,GAAG,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;QACjC,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;YACtB,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,CAAA;QACvB,CAAC;IACH,CAAC,CAAC,CAAA;AACJ,CAAC;AAED,MAAM,UAAU,cAAc,CAC5B,GAAY;IAEZ,IAAI,CAAC,GAAG;QAAE,OAAO,SAAS,CAAA;IAC1B,OAAO,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAA;AACzC,CAAC;AAQD,MAAM,CAAC,KAAK,UAAU,KAAK,CACzB,KAAkB,EAClB,UAAwB,EAAE;;IAE1B,MAAM,WAAW,GAAG,MAAA,OAAO,CAAC,QAAQ,0CAAE,KAAK,CAAC,GAAG,EAAE,GAAG,EAAE,CAAA;IACtD,MAAM,GAAG,GAAG,MAAA,OAAO,CAAC,GAAG,mCAAI,WAAW,CAAA;IACtC,MAAM,MAAM,GAAG,MAAA,OAAO,CAAC,MAAM,mCAAI,cAAc,CAAC,GAAG,CAAC,CAAA;IAEpD,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CACb,kBAAkB,GAAG,aAAH,GAAG,cAAH,GAAG,GAAI,IAAI,SAAS;YACpC,GAAG,SAAS,CAAC,IAAI,EAAE;SACpB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACf,CAAA;IACH,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,MAAM,EAAE,CAAA;IAC3B,OAAO,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;AAC7B,CAAC;AAED,eAAe;AACf,MAAM,UAAU,qBAAqB;IACnC,OAAO,CAAC,GAAG,SAAS,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAA;AAC/E,CAAC"}
@@ -0,0 +1,11 @@
1
+ import { Number2 } from '@hamster-note/types';
2
+ export interface DocumentAnchorWithTextId {
3
+ pageId: string;
4
+ textId?: string;
5
+ }
6
+ export interface DocumentAnchorWithPosition {
7
+ pageId: string;
8
+ position: Number2;
9
+ }
10
+ export type DocumentAnchor = DocumentAnchorWithPosition | DocumentAnchorWithTextId;
11
+ //# sourceMappingURL=DocumentAnchor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DocumentAnchor.d.ts","sourceRoot":"","sources":["../../src/types/DocumentAnchor.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,OAAO,EAAE,MAAM,qBAAqB,CAAA;AAE7C,MAAM,WAAW,wBAAwB;IACvC,MAAM,EAAE,MAAM,CAAA;IACd,MAAM,CAAC,EAAE,MAAM,CAAA;CAChB;AAGD,MAAM,WAAW,0BAA0B;IACzC,MAAM,EAAE,MAAM,CAAA;IACd,QAAQ,EAAE,OAAO,CAAA;CAClB;AAED,MAAM,MAAM,cAAc,GACtB,0BAA0B,GAC1B,wBAAwB,CAAA"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=DocumentAnchor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DocumentAnchor.js","sourceRoot":"","sources":["../../src/types/DocumentAnchor.ts"],"names":[],"mappings":""}
package/package.json CHANGED
@@ -1,13 +1,14 @@
1
1
  {
2
2
  "name": "@hamster-note/document-parser",
3
- "version": "0.1.0",
3
+ "version": "0.2.0",
4
4
  "description": "",
5
5
  "scripts": {
6
- "build:all": "echo 'no-build'",
6
+ "build:all": "tsc -p tsconfig.build.json",
7
+ "prepublishOnly": "yarn build:all",
7
8
  "test": "echo 'no-test'"
8
9
  },
9
10
  "dependencies": {
10
- "@hamster-note/types": "^0.1.0"
11
+ "@hamster-note/types": "^0.2.0"
11
12
  },
12
13
  "devDependencies": {
13
14
  "@eslint/js": "^9.39.2",
@@ -27,11 +28,18 @@
27
28
  "typescript": "^5.0.2",
28
29
  "typescript-eslint": "^8.50.0"
29
30
  },
30
- "main": "src/index.ts",
31
- "module": "src/index.ts",
32
- "types": "src/index.ts",
31
+ "type": "module",
32
+ "main": "./dist/index.js",
33
+ "module": "./dist/index.js",
34
+ "types": "./dist/index.d.ts",
35
+ "exports": {
36
+ ".": {
37
+ "types": "./dist/index.d.ts",
38
+ "import": "./dist/index.js"
39
+ }
40
+ },
33
41
  "files": [
34
- "src"
42
+ "dist"
35
43
  ],
36
44
  "packageManager": "yarn@1.22.22+sha512.a6b2f7906b721bba3d67d4aff083df04dad64c399707841b7acf00f6b133b7ac24255f2652fa22ae3534329dc6180534e98d17432037ff6fd140556e2bb3137e"
37
- }
45
+ }
@@ -1,13 +0,0 @@
1
- import { HamsterPage } from '../Page'
2
- import { IntermediateOutline } from '@hamster-note/types'
3
-
4
- export abstract class HamsterDocument {
5
- // 获取 Page 对象列表
6
- abstract getPages(): Promise<HamsterPage[]>
7
- // 获取 Page 对象
8
- abstract getPage(pageNumber: number): Promise<HamsterPage | undefined>
9
- // 获取大纲(PDF如果有内置的话)
10
- abstract getOutline(): Promise<IntermediateOutline | undefined>
11
- // 获取封面
12
- abstract getCover(): Promise<HTMLCanvasElement | HTMLImageElement>
13
- }
package/src/Page/index.ts DELETED
@@ -1,22 +0,0 @@
1
- import { Number2 } from '@hamster-note/types'
2
-
3
- export enum RenderViews {
4
- THUMBNAIL = 'thumbnail',
5
- TEXT = 'text'
6
- }
7
-
8
- export interface RenderOptions {
9
- views: RenderViews[]
10
- scale: number
11
- }
12
-
13
- export abstract class HamsterPage {
14
- abstract render(
15
- container: HTMLDivElement,
16
- options?: RenderOptions
17
- ): Promise<void>
18
- abstract getNumber(): number
19
- abstract getSize(scale: number): Number2
20
- abstract getPureText(): string
21
- // abstract getTextDom
22
- }
@@ -1,28 +0,0 @@
1
- import { HamsterDocument } from '../Document'
2
- import { IntermediateDocument } from '@hamster-note/types'
3
-
4
- export class DocumentParser {
5
- static readonly ext: string
6
- // Base static methods to be overridden by concrete parsers
7
- static async encode(
8
- _file: File | ArrayBuffer
9
- ): Promise<IntermediateDocument | HamsterDocument | undefined> {
10
- return Promise.resolve(undefined)
11
- }
12
- static async decode(
13
- _intermediateDocument: IntermediateDocument
14
- ): Promise<File | ArrayBuffer | undefined> {
15
- return Promise.resolve(undefined)
16
- }
17
- protected static async toArrayBuffer(
18
- fileOrBuffer: File | ArrayBuffer
19
- ): Promise<ArrayBuffer> {
20
- if (fileOrBuffer instanceof ArrayBuffer) return fileOrBuffer
21
- return await new Promise<ArrayBuffer>((resolve, reject) => {
22
- const reader = new FileReader()
23
- reader.readAsArrayBuffer(fileOrBuffer)
24
- reader.onload = () => resolve(reader.result as ArrayBuffer)
25
- reader.onerror = (e) => reject(e)
26
- })
27
- }
28
- }
package/src/index.ts DELETED
@@ -1,5 +0,0 @@
1
- export * from './Document'
2
- export * from './Page'
3
- export * from './types/DocumentAnchor'
4
- export * from './Parser'
5
- export * from './register'
@@ -1,30 +0,0 @@
1
- import { DocumentParser } from '../Parser'
2
-
3
- // Store constructors (classes), not instances
4
- const parserMap: Map<string, typeof DocumentParser> = new Map()
5
-
6
- // Register a parser class (constructor) which exposes a static `ext`
7
- export const registerParser = (parser: typeof DocumentParser) => {
8
- console.log(parser.ext, parser)
9
- parserMap.set(parser.ext, parser)
10
- }
11
-
12
- export function getParser(file: File): typeof DocumentParser | undefined {
13
- const extension = file.name.split('.').pop()
14
- if (!extension) {
15
- return undefined
16
- }
17
- return parserMap.get(extension)
18
- }
19
-
20
- export function parse(file: File) {
21
- const Parser = getParser(file)
22
- console.log('Parse:', Parser)
23
- if (!Parser) {
24
- return undefined
25
- }
26
- // call static encode on the parser class
27
- const res = Parser.encode(file)
28
- // res.then((_doc) => { /* window.doc1 = _doc */ })
29
- return res
30
- }
@@ -1,17 +0,0 @@
1
- // 文档锚(标记了文字id)
2
- import { Number2 } from '@hamster-note/types'
3
-
4
- export interface DocumentAnchorWithTextId {
5
- pageId: string
6
- textId?: string
7
- }
8
-
9
- // 文档锚(标记了位置)
10
- export interface DocumentAnchorWithPosition {
11
- pageId: string
12
- position: Number2
13
- }
14
-
15
- export type DocumentAnchor =
16
- | DocumentAnchorWithPosition
17
- | DocumentAnchorWithTextId