@wingleeio/ori-react 0.1.0 → 0.1.2

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.
package/dist/index.cjs CHANGED
@@ -151,6 +151,16 @@ var EditorView = class {
151
151
  on("beforeinput", (e) => this.onBeforeInput(e));
152
152
  on("input", () => this.onInput());
153
153
  on("keydown", (e) => this.onKeyDown(e));
154
+ on("blur", () => {
155
+ setTimeout(() => {
156
+ if (document.activeElement === this.root || !document.hasFocus()) return;
157
+ const sel = this.editor.getSelection();
158
+ if (sel && !oriCore.isCollapsed(sel)) {
159
+ this.editor.collapse(sel.focus);
160
+ this.lastRevision = this.rev();
161
+ }
162
+ }, 0);
163
+ });
154
164
  on("compositionstart", () => this.composing = true);
155
165
  on("compositionend", () => {
156
166
  this.composing = false;
@@ -467,11 +477,21 @@ function caretClientRect() {
467
477
  const rects = r.getClientRects();
468
478
  if (rects.length) return rects[rects.length - 1];
469
479
  const b = r.getBoundingClientRect();
470
- return b.height || b.width ? b : null;
480
+ if (b.height || b.width) return b;
481
+ const node = r.startContainer;
482
+ const el = (node.nodeType === Node.TEXT_NODE ? node.parentElement : node) ?? null;
483
+ if (!el) return null;
484
+ const eb = el.getBoundingClientRect();
485
+ const cs = getComputedStyle(el);
486
+ const lh = parseFloat(cs.lineHeight) || parseFloat(cs.fontSize) * 1.4 || 18;
487
+ const padL = parseFloat(cs.paddingLeft) || 0;
488
+ const padT = parseFloat(cs.paddingTop) || 0;
489
+ return new DOMRect(eb.left + padL, eb.top + padT, 0, lh);
471
490
  }
472
491
  var NoteEditor = react.forwardRef(function NoteEditor2({ editor, className, style, maxWidth = 720, placeholder, autoFocus, readOnly, blockRenderers, atomRenderers }, ref) {
473
492
  const snapshot = useEditorSnapshot(editor);
474
493
  const scrollerRef = react.useRef(null);
494
+ const overlayRef = react.useRef(null);
475
495
  const contentRef = react.useRef(null);
476
496
  const viewRef = react.useRef(null);
477
497
  const [focused, setFocused] = react.useState(false);
@@ -493,7 +513,8 @@ var NoteEditor = react.forwardRef(function NoteEditor2({ editor, className, styl
493
513
  if (!b.width && !b.height) return null;
494
514
  return { top: b.top, left: b.left, right: b.right, bottom: b.bottom, width: b.width, height: b.height };
495
515
  },
496
- getScrollElement: () => scrollerRef.current
516
+ getScrollElement: () => scrollerRef.current,
517
+ getOverlayElement: () => overlayRef.current
497
518
  }),
498
519
  []
499
520
  );
@@ -556,8 +577,28 @@ var NoteEditor = react.forwardRef(function NoteEditor2({ editor, className, styl
556
577
  const sc = scrollerRef.current;
557
578
  if (sc) editor.setViewport(sc.scrollTop, sc.clientHeight);
558
579
  };
580
+ const onPointerDown = (e) => {
581
+ if (readOnly) return;
582
+ const t = e.target;
583
+ const onSurface = t === scrollerRef.current || t === overlayRef.current || t.classList.contains("ori-ce") || t.dataset.spacer != null;
584
+ if (!onSurface) return;
585
+ const content = contentRef.current;
586
+ const blocks = content?.querySelectorAll("[data-block-id]");
587
+ const last = blocks && blocks[blocks.length - 1];
588
+ if (!content || !last) return;
589
+ if (e.clientY <= last.getBoundingClientRect().bottom) return;
590
+ e.preventDefault();
591
+ content.focus();
592
+ const sel = window.getSelection();
593
+ if (!sel) return;
594
+ const range = document.createRange();
595
+ range.selectNodeContents(last);
596
+ range.collapse(false);
597
+ sel.removeAllRanges();
598
+ sel.addRange(range);
599
+ };
559
600
  const showCaret = focused && !!caret && !readOnly;
560
- return /* @__PURE__ */ jsxRuntime.jsx("div", { className: `ori-root${className ? ` ${className}` : ""}`, style, children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "ori-scroller", ref: scrollerRef, onScroll, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "ori-content", style: { maxWidth, marginInline: "auto", position: "relative" }, children: [
601
+ return /* @__PURE__ */ jsxRuntime.jsx("div", { className: `ori-root${className ? ` ${className}` : ""}`, style, children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: "ori-scroller", ref: scrollerRef, onScroll, onPointerDown, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "ori-content", ref: overlayRef, style: { maxWidth, marginInline: "auto", position: "relative" }, children: [
561
602
  /* @__PURE__ */ jsxRuntime.jsx(
562
603
  "div",
563
604
  {
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/useEditor.ts","../src/hooks.ts","../src/ce/dom.ts","../src/ce/view.ts","../src/NoteEditor.tsx","../src/renderers.tsx"],"names":["useRef","EditorController","createCanvasMeasurer","useEffect","useSyncExternalStore","textNode","createRoot","isCollapsed","forwardRef","NoteEditor","useState","useImperativeHandle","useLayoutEffect","jsx","jsxs","createContext","useContext"],"mappings":";;;;;;;;;;AA8BO,SAAS,SAAA,CAAU,OAAA,GAA4B,EAAC,EAAqB;AAC1E,EAAA,MAAM,GAAA,GAAMA,aAAgC,IAAI,CAAA;AAChD,EAAA,IAAI,GAAA,CAAI,YAAY,IAAA,EAAM;AACxB,IAAA,GAAA,CAAI,OAAA,GAAU,IAAIC,wBAAA,CAAiB;AAAA,MACjC,KAAK,OAAA,CAAQ,GAAA;AAAA,MACb,QAAA,EAAU,OAAA,CAAQ,QAAA,IAAYC,4BAAA,EAAqB;AAAA,MACnD,YAAY,OAAA,CAAQ,UAAA;AAAA,MACpB,UAAU,OAAA,CAAQ,QAAA;AAAA,MAClB,cAAc,OAAA,CAAQ,YAAA;AAAA,MACtB,QAAQ,OAAA,CAAQ;AAAA,KACjB,CAAA;AAAA,EACH;AACA,EAAAC,eAAA,CAAU,MAAM;AACd,IAAA,MAAM,SAAS,GAAA,CAAI,OAAA;AACnB,IAAA,MAAA,EAAQ,OAAA,EAAQ;AAChB,IAAA,OAAO,MAAM,QAAQ,UAAA,EAAW;AAAA,EAClC,CAAA,EAAG,EAAE,CAAA;AACL,EAAA,OAAO,GAAA,CAAI,OAAA;AACb;AC5CO,SAAS,kBAAkB,MAAA,EAA0C;AAC1E,EAAA,OAAOC,2BAAqB,MAAA,CAAO,SAAA,EAAW,MAAA,CAAO,WAAA,EAAa,OAAO,WAAW,CAAA;AACtF;AAOO,SAAS,eAAe,MAAA,EAAiC;AAC9D,EAAA,MAAM,QAAA,GAAW,kBAAkB,MAAM,CAAA;AAEzC,EAAA,KAAK,QAAA,CAAS,QAAA;AACd,EAAA,OAAO,OAAO,cAAA,EAAe;AAC/B;;;ACfO,SAAS,IAAI,CAAA,EAAmB;AACrC,EAAA,OAAO,OAAO,GAAA,KAAQ,WAAA,IAAe,GAAA,CAAI,MAAA,GAAS,GAAA,CAAI,MAAA,CAAO,CAAC,CAAA,GAAI,CAAA,CAAE,OAAA,CAAQ,QAAA,EAAU,MAAM,CAAA;AAC9F;AAWO,SAAS,SAAA,CAAU,MAAmB,IAAA,EAAuC;AAClF,EAAA,IAAI,CAAA,GAAiB,IAAA;AACrB,EAAA,OAAO,CAAA,IAAK,MAAM,IAAA,EAAM;AACtB,IAAA,IAAI,CAAA,YAAa,WAAA,IAAe,CAAA,CAAE,OAAA,CAAQ,SAAS,OAAO,CAAA;AAC1D,IAAA,CAAA,GAAI,CAAA,CAAE,UAAA;AAAA,EACR;AACA,EAAA,OAAO,IAAA;AACT;AAEA,SAAS,OAAO,IAAA,EAAuC;AACrD,EAAA,IAAI,CAAA,GAAiB,IAAA;AACrB,EAAA,OAAO,CAAA,EAAG;AACR,IAAA,IAAI,aAAa,WAAA,IAAe,CAAA,CAAE,OAAA,CAAQ,GAAA,IAAO,MAAM,OAAO,CAAA;AAC9D,IAAA,CAAA,GAAI,CAAA,CAAE,UAAA;AAAA,EACR;AACA,EAAA,OAAO,IAAA;AACT;AAGO,SAAS,UAAA,CACd,IAAA,EACA,IAAA,EACA,MAAA,EAC4C;AAC5C,EAAA,MAAM,OAAA,GAAU,SAAA,CAAU,IAAA,EAAM,IAAI,CAAA;AACpC,EAAA,IAAI,CAAC,SAAS,OAAO,IAAA;AACrB,EAAA,MAAM,OAAA,GAAU,QAAQ,OAAA,CAAQ,OAAA;AAEhC,EAAA,IAAI,IAAA,IAAQ,IAAA,CAAK,QAAA,KAAa,IAAA,CAAK,SAAA,EAAW;AAC5C,IAAA,MAAM,IAAA,GAAO,OAAO,IAAI,CAAA;AACxB,IAAA,MAAM,OAAO,IAAA,GAAO,MAAA,CAAO,IAAA,CAAK,OAAA,CAAQ,GAAG,CAAA,GAAI,CAAA;AAC/C,IAAA,OAAO,EAAE,OAAA,EAAS,MAAA,EAAQ,IAAA,GAAO,MAAA,EAAO;AAAA,EAC1C;AAGA,EAAA,MAAM,EAAA,GAAK,IAAA;AACX,EAAA,IAAI,EAAA,CAAG,OAAA,EAAS,GAAA,IAAO,IAAA,EAAM;AAE3B,IAAA,OAAO,EAAE,OAAA,EAAS,MAAA,EAAQ,MAAA,CAAO,EAAA,CAAG,OAAA,CAAQ,GAAG,CAAA,IAAK,MAAA,GAAS,CAAA,GAAI,OAAA,CAAQ,EAAE,IAAI,CAAA,CAAA,EAAG;AAAA,EACpF;AACA,EAAA,MAAM,IAAA,GAAO,KAAA,CAAM,IAAA,CAAK,EAAA,CAAG,UAAU,CAAA;AACrC,EAAA,KAAA,IAAS,CAAA,GAAI,MAAA,EAAQ,CAAA,GAAI,IAAA,CAAK,QAAQ,CAAA,EAAA,EAAK;AACzC,IAAA,MAAM,CAAA,GAAI,KAAK,CAAC,CAAA;AAChB,IAAA,IAAI,CAAA,YAAa,WAAA,IAAe,CAAA,CAAE,OAAA,CAAQ,OAAO,IAAA,EAAM,OAAO,EAAE,OAAA,EAAS,MAAA,EAAQ,MAAA,CAAO,CAAA,CAAE,OAAA,CAAQ,GAAG,CAAA,EAAE;AAAA,EACzG;AAEA,EAAA,IAAI,GAAA,GAAM,CAAA;AACV,EAAA,KAAA,MAAW,CAAA,IAAK,MAAM,IAAI,CAAA,YAAa,eAAe,CAAA,CAAE,OAAA,CAAQ,OAAO,IAAA,EAAM,GAAA,GAAM,KAAK,GAAA,CAAI,GAAA,EAAK,OAAO,CAAA,CAAE,OAAA,CAAQ,GAAG,CAAA,GAAI,OAAA,CAAQ,CAAC,CAAC,CAAA;AACnI,EAAA,OAAO,EAAE,OAAA,EAAS,MAAA,EAAQ,GAAA,EAAI;AAChC;AAEA,SAAS,QAAQ,IAAA,EAA2B;AAC1C,EAAA,OAAO,IAAA,CAAK,OAAA,CAAQ,GAAA,IAAO,IAAA,GAAO,MAAA,CAAO,IAAA,CAAK,OAAA,CAAQ,GAAG,CAAA,GAAA,CAAK,IAAA,CAAK,WAAA,IAAe,EAAA,EAAI,MAAA;AACxF;AAGO,SAAS,UAAA,CACd,IAAA,EACA,OAAA,EACA,MAAA,EACuC;AACvC,EAAA,MAAM,UAAU,IAAA,CAAK,aAAA,CAAc,mBAAmB,GAAA,CAAI,OAAO,CAAC,CAAA,EAAA,CAAI,CAAA;AACtE,EAAA,IAAI,CAAC,SAAS,OAAO,IAAA;AACrB,EAAA,MAAM,QAAQ,KAAA,CAAM,IAAA,CAAK,OAAA,CAAQ,gBAAA,CAAiB,YAAY,CAAC,CAAA;AAC/D,EAAA,IAAI,KAAA,CAAM,WAAW,CAAA,EAAG;AACtB,IAAA,OAAO,EAAE,IAAA,EAAM,OAAA,EAAS,MAAA,EAAQ,CAAA,EAAE;AAAA,EACpC;AACA,EAAA,KAAA,MAAW,QAAQ,KAAA,EAAO;AACxB,IAAA,MAAM,KAAA,GAAQ,MAAA,CAAO,IAAA,CAAK,OAAA,CAAQ,GAAG,CAAA;AACrC,IAAA,MAAM,GAAA,GAAM,QAAQ,IAAI,CAAA;AACxB,IAAA,IAAI,MAAA,IAAU,QAAQ,GAAA,EAAK;AACzB,MAAA,IAAI,IAAA,CAAK,OAAA,CAAQ,IAAA,IAAQ,IAAA,EAAM;AAE7B,QAAA,MAAM,MAAM,KAAA,CAAM,SAAA,CAAU,QAAQ,IAAA,CAAK,OAAA,CAAQ,YAAY,IAAI,CAAA;AACjE,QAAA,OAAO,EAAE,MAAM,OAAA,EAAS,MAAA,EAAQ,UAAU,KAAA,GAAQ,GAAA,GAAM,MAAM,CAAA,EAAE;AAAA,MAClE;AACA,MAAA,MAAMC,SAAAA,GAAW,KAAK,UAAA,IAAc,IAAA;AACpC,MAAA,OAAO,EAAE,IAAA,EAAMA,SAAAA,EAAU,MAAA,EAAQ,IAAA,CAAK,IAAI,CAAA,EAAG,IAAA,CAAK,GAAA,CAAI,MAAA,GAAS,QAAQA,SAAAA,CAAS,WAAA,IAAe,EAAA,EAAI,MAAM,CAAC,CAAA,EAAE;AAAA,IAC9G;AAAA,EACF;AAEA,EAAA,MAAM,IAAA,GAAO,KAAA,CAAM,KAAA,CAAM,MAAA,GAAS,CAAC,CAAA;AACnC,EAAA,MAAM,QAAA,GAAW,KAAK,UAAA,IAAc,IAAA;AACpC,EAAA,OAAO,EAAE,IAAA,EAAM,QAAA,EAAU,SAAS,QAAA,CAAS,WAAA,IAAe,IAAI,MAAA,EAAO;AACvE;AAEA,SAAS,UAAU,KAAA,EAAoC;AACrD,EAAA,MAAM,CAAA,GAAI,SAAS,EAAC;AACpB,EAAA,MAAM,GAAA,GAAM,CAAC,UAAU,CAAA;AACvB,EAAA,IAAI,CAAA,CAAE,IAAA,EAAM,GAAA,CAAI,IAAA,CAAK,YAAY,CAAA;AACjC,EAAA,IAAI,CAAA,CAAE,MAAA,EAAQ,GAAA,CAAI,IAAA,CAAK,cAAc,CAAA;AACrC,EAAA,IAAI,CAAA,CAAE,SAAA,EAAW,GAAA,CAAI,IAAA,CAAK,iBAAiB,CAAA;AAC3C,EAAA,IAAI,CAAA,CAAE,MAAA,EAAQ,GAAA,CAAI,IAAA,CAAK,cAAc,CAAA;AACrC,EAAA,IAAI,CAAA,CAAE,IAAA,EAAM,GAAA,CAAI,IAAA,CAAK,eAAe,CAAA;AACpC,EAAA,IAAI,CAAA,CAAE,IAAA,EAAM,GAAA,CAAI,IAAA,CAAK,eAAe,CAAA;AACpC,EAAA,OAAO,GAAA,CAAI,KAAK,GAAG,CAAA;AACrB;AAGO,SAAS,SAAS,IAAA,EAA+B;AACtD,EAAA,MAAM,IAAA,GAAO,QAAA,CAAS,aAAA,CAAc,MAAM,CAAA;AAC1C,EAAA,IAAA,CAAK,SAAA,GAAY,SAAA,CAAU,IAAA,CAAK,KAAK,CAAA;AACrC,EAAA,IAAA,CAAK,OAAA,CAAQ,GAAA,GAAM,MAAA,CAAO,IAAA,CAAK,KAAK,CAAA;AACpC,EAAA,IAAA,CAAK,OAAA,CAAQ,GAAA,GAAM,MAAA,CAAO,IAAA,CAAK,KAAK,MAAM,CAAA;AAC1C,EAAA,IAAA,CAAK,cAAc,IAAA,CAAK,IAAA;AACxB,EAAA,OAAO,IAAA;AACT;;;ACnHA,IAAM,WAAA,GAAc,QAAA;AAeb,IAAM,aAAN,MAAiB;AAAA,EAStB,WAAA,CACU,IAAA,EACA,MAAA,EACA,IAAA,EACR;AAHQ,IAAA,aAAA,CAAA,IAAA,EAAA,MAAA,EAAA,IAAA,CAAA;AACA,IAAA,aAAA,CAAA,IAAA,EAAA,QAAA,EAAA,MAAA,CAAA;AACA,IAAA,aAAA,CAAA,IAAA,EAAA,MAAA,EAAA,IAAA,CAAA;AAXV,IAAA,aAAA,CAAA,IAAA,EAAQ,OAAA,sBAAY,GAAA,EAAuB,CAAA;AAC3C,IAAA,aAAA,CAAA,IAAA,EAAQ,WAAA,EAAY,KAAA,CAAA;AACpB,IAAA,aAAA,CAAA,IAAA,EAAQ,eAAA,EAAgB,KAAA,CAAA;AACxB,IAAA,aAAA,CAAA,IAAA,EAAQ,aAA+B,EAAC,CAAA;AAGxC;AAAA;AAAA,IAAA,aAAA,CAAA,IAAA,EAAQ,cAAA,EAAe,EAAA,CAAA;AAOrB,IAAA,IAAA,CAAK,YAAA,CAAa,iBAAA,EAAmB,IAAA,CAAK,QAAA,GAAW,UAAU,MAAM,CAAA;AACrE,IAAA,IAAA,CAAK,YAAA,CAAa,YAAA,EAAc,IAAA,CAAK,QAAA,GAAW,UAAU,MAAM,CAAA;AAChE,IAAA,IAAA,CAAK,YAAA,CAAa,QAAQ,SAAS,CAAA;AACnC,IAAA,IAAA,CAAK,YAAA,CAAa,kBAAkB,MAAM,CAAA;AAC1C,IAAA,IAAA,CAAK,YAAA,EAAa;AAClB,IAAA,IAAA,CAAK,YAAA,GAAe,KAAK,GAAA,EAAI;AAE7B,IAAA,MAAM,EAAA,GAAK,CACT,CAAA,EACA,CAAA,EACA,CAAA,KACG;AACH,MAAA,IAAA,CAAK,gBAAA,CAAiB,CAAA,EAAG,CAAA,EAAoB,CAAC,CAAA;AAC9C,MAAA,IAAA,CAAK,SAAA,CAAU,KAAK,MAAM,IAAA,CAAK,oBAAoB,CAAA,EAAG,CAAA,EAAoB,CAAC,CAAC,CAAA;AAAA,IAC9E,CAAA;AACA,IAAA,EAAA,CAAG,eAAe,CAAC,CAAA,KAAM,IAAA,CAAK,aAAA,CAAc,CAAe,CAAC,CAAA;AAC5D,IAAA,EAAA,CAAG,OAAA,EAAS,MAAM,IAAA,CAAK,OAAA,EAAS,CAAA;AAChC,IAAA,EAAA,CAAG,WAAW,CAAC,CAAA,KAAM,IAAA,CAAK,SAAA,CAAU,CAAkB,CAAC,CAAA;AACvD,IAAA,EAAA,CAAG,kBAAA,EAAoB,MAAO,IAAA,CAAK,SAAA,GAAY,IAAK,CAAA;AACpD,IAAA,EAAA,CAAG,kBAAkB,MAAM;AACzB,MAAA,IAAA,CAAK,SAAA,GAAY,KAAA;AACjB,MAAA,IAAA,CAAK,OAAA,EAAQ;AAAA,IACf,CAAC,CAAA;AAED,IAAA,MAAM,cAAc,MAAM;AACxB,MAAA,IAAI,IAAA,CAAK,aAAA,IAAiB,IAAA,CAAK,SAAA,EAAW;AAC1C,MAAA,MAAM,GAAA,GAAM,KAAK,aAAA,EAAc;AAC/B,MAAA,IAAI,CAAC,GAAA,EAAK;AACV,MAAA,IAAA,CAAK,MAAA,CAAO,aAAa,GAAG,CAAA;AAG5B,MAAA,IAAA,CAAK,YAAA,GAAe,KAAK,GAAA,EAAI;AAAA,IAC/B,CAAA;AACA,IAAA,QAAA,CAAS,gBAAA,CAAiB,mBAAmB,WAAW,CAAA;AACxD,IAAA,IAAA,CAAK,UAAU,IAAA,CAAK,MAAM,SAAS,mBAAA,CAAoB,iBAAA,EAAmB,WAAW,CAAC,CAAA;AAAA,EACxF;AAAA,EAEA,OAAA,GAAU;AACR,IAAA,IAAA,CAAK,SAAA,CAAU,OAAA,CAAQ,CAAC,CAAA,KAAM,GAAG,CAAA;AACjC,IAAA,IAAA,CAAK,MAAM,OAAA,CAAQ,CAAC,CAAA,KAAM,CAAA,CAAE,SAAS,CAAA;AACrC,IAAA,IAAA,CAAK,MAAM,KAAA,EAAM;AAAA,EACnB;AAAA,EAEA,KAAA,GAAQ;AACN,IAAA,IAAA,CAAK,KAAK,KAAA,EAAM;AAAA,EAClB;AAAA;AAAA,EAIQ,GAAA,GAAc;AACpB,IAAA,OAAO,IAAA,CAAK,MAAA,CAAO,WAAA,EAAY,CAAE,QAAA;AAAA,EACnC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAA,GAAO;AACL,IAAA,MAAM,GAAA,GAAM,KAAK,GAAA,EAAI;AACrB,IAAA,IAAI,GAAA,KAAQ,KAAK,YAAA,EAAc;AAI/B,IAAA,MAAM,OAAA,GAAU,KAAK,YAAA,EAAa;AAClC,IAAA,IAAI,OAAA,OAAc,cAAA,EAAe;AACjC,IAAA,IAAA,CAAK,YAAA,GAAe,GAAA;AAAA,EACtB;AAAA;AAAA,EAGQ,MAAA,GAAS;AACf,IAAA,IAAA,CAAK,YAAA,EAAa;AAClB,IAAA,IAAA,CAAK,cAAA,EAAe;AACpB,IAAA,IAAA,CAAK,YAAA,GAAe,KAAK,GAAA,EAAI;AAAA,EAC/B;AAAA;AAAA,EAGQ,IAAI,EAAA,EAAoB;AAC9B,IAAA,OAAO,IAAA,CAAK,MAAA,CAAO,YAAA,CAAa,EAAE,CAAA,GAAI,GAAA,GAAM,IAAA,CAAK,SAAA,CAAU,IAAA,CAAK,MAAA,CAAO,SAAA,CAAU,EAAE,CAAC,CAAA;AAAA,EACtF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQQ,YAAA,GAAwB;AAC9B,IAAA,IAAI,OAAA,GAAU,KAAA;AACd,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,MAAA,CAAO,WAAA,EAAY;AACrC,IAAA,MAAM,MAAM,IAAA,CAAK,OAAA;AACjB,IAAA,MAAM,OAAO,GAAA,CAAI,MAAA,GAAS,GAAA,CAAI,CAAC,EAAE,GAAA,GAAM,CAAA;AACvC,IAAA,MAAM,IAAA,GAAO,GAAA,CAAI,MAAA,GACb,IAAA,CAAK,GAAA,CAAI,GAAG,IAAA,CAAK,WAAA,IAAe,GAAA,CAAI,GAAA,CAAI,MAAA,GAAS,CAAC,EAAE,GAAA,GAAM,GAAA,CAAI,GAAA,CAAI,MAAA,GAAS,CAAC,CAAA,CAAE,MAAA,CAAO,CAAA,GACrF,IAAA,CAAK,GAAA,CAAI,CAAA,EAAG,IAAA,CAAK,WAAW,CAAA;AAEhC,IAAA,MAAM,GAAA,GAAM,OAAA;AACZ,IAAA,MAAM,MAAA,GAAS,UAAA;AACf,IAAA,MAAM,IAAA,GAAO,CAAC,GAAA,EAAK,GAAG,GAAA,CAAI,GAAA,CAAI,CAAC,CAAA,KAAM,CAAA,CAAE,EAAE,CAAA,EAAG,MAAM,CAAA;AAClD,IAAA,MAAM,KAAA,GAAQ,CAAC,EAAA,KAAgB;AAC7B,MAAA,MAAM,CAAA,GAAI,EAAA;AACV,MAAA,OAAO,CAAA,CAAE,QAAQ,MAAA,GAAS,IAAA,GAAM,EAAE,OAAA,CAAQ,MAAA,GAAU,CAAA,CAAE,OAAA,CAAQ,OAAA,IAAW,EAAA;AAAA,IAC3E,CAAA;AACA,IAAA,MAAM,IAAA,uBAAW,GAAA,EAAyB;AAC1C,IAAA,KAAA,MAAW,CAAA,IAAK,KAAA,CAAM,IAAA,CAAK,IAAA,CAAK,IAAA,CAAK,QAAQ,CAAA,EAAG,IAAA,CAAK,GAAA,CAAI,KAAA,CAAM,CAAC,CAAA,EAAG,CAAgB,CAAA;AAEnF,IAAA,IAAI,IAAA,GAA2B,IAAA;AAC/B,IAAA,KAAA,MAAW,KAAK,IAAA,EAAM;AACpB,MAAA,IAAI,EAAA,GAA8B,IAAA,CAAK,GAAA,CAAI,CAAC,CAAA;AAC5C,MAAA,IAAI,EAAA,EAAI;AACN,QAAA,IAAA,CAAK,OAAO,CAAC,CAAA;AAAA,MACf,CAAA,MAAO;AACL,QAAA,EAAA,GAAK,CAAA,KAAM,GAAA,IAAO,CAAA,KAAM,MAAA,GAAS,IAAA,CAAK,UAAA,CAAW,CAAA,CAAE,KAAA,CAAM,CAAC,CAAC,CAAA,GAAI,IAAA,CAAK,UAAU,CAAC,CAAA;AAC/E,QAAA,OAAA,GAAU,IAAA;AAAA,MACZ;AACA,MAAA,MAAM,MAAA,GAA2B,IAAA,GAAO,IAAA,CAAK,WAAA,GAAc,KAAK,IAAA,CAAK,UAAA;AACrE,MAAA,IAAI,WAAW,EAAA,EAAI;AACjB,QAAA,IAAA,CAAK,IAAA,CAAK,YAAA,CAAa,EAAA,EAAI,MAAM,CAAA;AACjC,QAAA,OAAA,GAAU,IAAA;AAAA,MACZ;AACA,MAAA,IAAA,GAAO,EAAA;AAAA,IACT;AACA,IAAA,KAAA,MAAW,EAAA,IAAM,IAAA,CAAK,MAAA,EAAO,EAAG;AAC9B,MAAA,IAAI,EAAA,CAAG,OAAA,CAAQ,OAAA,EAAS,IAAA,CAAK,eAAe,EAAE,CAAA;AAC9C,MAAA,EAAA,CAAG,MAAA,EAAO;AACV,MAAA,OAAA,GAAU,IAAA;AAAA,IACZ;AAEA,IAAA,MAAM,GAAA,GAAM,KAAK,IAAA,CAAK,iBAAA;AACtB,IAAA,IAAI,GAAA,IAAO,GAAA,CAAI,KAAA,CAAM,MAAA,KAAW,CAAA,EAAG,IAAI,CAAA,EAAA,CAAA,EAAM,GAAA,CAAI,KAAA,CAAM,MAAA,GAAS,CAAA,EAAG,IAAI,CAAA,EAAA,CAAA;AACvE,IAAA,MAAM,GAAA,GAAM,KAAK,IAAA,CAAK,gBAAA;AACtB,IAAA,IAAI,GAAA,IAAO,GAAA,CAAI,KAAA,CAAM,MAAA,KAAW,CAAA,EAAG,IAAI,CAAA,EAAA,CAAA,EAAM,GAAA,CAAI,KAAA,CAAM,MAAA,GAAS,CAAA,EAAG,IAAI,CAAA,EAAA,CAAA;AAEvE,IAAA,KAAA,MAAW,MAAM,GAAA,EAAK;AACpB,MAAA,MAAM,EAAA,GAAK,KAAK,IAAA,CAAK,aAAA,CAAc,mBAAmB,GAAA,CAAI,EAAA,CAAG,EAAE,CAAC,CAAA,EAAA,CAAI,CAAA;AACpE,MAAA,IAAI,CAAC,EAAA,EAAI;AACT,MAAA,MAAM,GAAA,GAAM,IAAA,CAAK,GAAA,CAAI,EAAA,CAAG,EAAE,CAAA;AAC1B,MAAA,IAAI,EAAA,CAAG,OAAA,CAAQ,GAAA,KAAQ,GAAA,EAAK;AAC1B,QAAA,EAAA,CAAG,QAAQ,GAAA,GAAM,GAAA;AACjB,QAAA,IAAA,CAAK,gBAAA,CAAiB,EAAA,EAAI,EAAA,CAAG,EAAE,CAAA;AAC/B,QAAA,OAAA,GAAU,IAAA;AAAA,MACZ;AAAA,IACF;AACA,IAAA,OAAO,OAAA;AAAA,EACT;AAAA,EAEQ,UAAU,EAAA,EAAyB;AACzC,IAAA,MAAM,EAAA,GAAK,QAAA,CAAS,aAAA,CAAc,KAAK,CAAA;AACvC,IAAA,EAAA,CAAG,QAAQ,OAAA,GAAU,EAAA;AACrB,IAAA,OAAO,EAAA;AAAA,EACT;AAAA,EAEQ,WAAW,KAAA,EAA4B;AAC7C,IAAA,MAAM,EAAA,GAAK,QAAA,CAAS,aAAA,CAAc,KAAK,CAAA;AACvC,IAAA,EAAA,CAAG,QAAQ,MAAA,GAAS,KAAA;AACpB,IAAA,EAAA,CAAG,YAAA,CAAa,mBAAmB,OAAO,CAAA;AAC1C,IAAA,EAAA,CAAG,YAAA,CAAa,eAAe,MAAM,CAAA;AACrC,IAAA,EAAA,CAAG,MAAM,UAAA,GAAa,MAAA;AACtB,IAAA,EAAA,CAAG,MAAM,aAAA,GAAgB,MAAA;AACzB,IAAA,OAAO,EAAA;AAAA,EACT;AAAA,EAEQ,gBAAA,CAAiB,IAAiB,EAAA,EAAY;AACpD,IAAA,IAAA,CAAK,eAAe,EAAE,CAAA;AACtB,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,MAAA,CAAO,YAAA,CAAa,EAAE,CAAA;AACxC,IAAA,EAAA,CAAG,SAAA,GAAY,uBAAuB,IAAI,CAAA,CAAA;AAE1C,IAAA,MAAM,aAAA,GAAgB,IAAA,CAAK,IAAA,CAAK,WAAA,CAAY,IAAI,CAAA;AAChD,IAAA,IAAI,aAAA,EAAe;AACjB,MAAA,EAAA,CAAG,eAAA,GAAkB,OAAA;AACrB,MAAA,EAAA,CAAG,WAAA,GAAc,EAAA;AACjB,MAAA,MAAM,IAAA,GAAOC,kBAAW,EAAE,CAAA;AAC1B,MAAA,IAAA,CAAK,MAAA,CAAO,aAAA,CAAc,EAAE,MAAA,EAAQ,IAAA,CAAK,QAAQ,KAAA,EAAO,EAAE,EAAA,EAAI,IAAA,EAAM,KAAA,EAAO,CAAA,EAAG,KAAK,CAAA,EAAG,MAAA,EAAQ,CAAA,EAAE,EAAG,MAAA,EAAQ,IAAA,CAAK,OAAO,SAAA,CAAU,EAAE,CAAA,EAAI,CAAc,CAAA;AACrJ,MAAA,IAAA,CAAK,KAAA,CAAM,GAAA,CAAI,EAAA,EAAI,IAAI,CAAA;AACvB,MAAA;AAAA,IACF;AACA,IAAA,EAAA,CAAG,eAAA,GAAkB,SAAA;AACrB,IAAA,EAAA,CAAG,WAAA,GAAc,EAAA;AACjB,IAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,MAAA,CAAO,SAAA,CAAU,EAAE,CAAA;AACtC,IAAA,IAAI,KAAA,CAAM,WAAW,CAAA,EAAG;AACtB,MAAA,EAAA,CAAG,WAAA,CAAY,QAAA,CAAS,aAAA,CAAc,IAAI,CAAC,CAAA;AAC3C,MAAA;AAAA,IACF;AACA,IAAA,KAAA,MAAW,QAAQ,KAAA,EAAO;AACxB,MAAA,IAAI,KAAK,IAAA,EAAM;AACb,QAAA,MAAM,IAAA,GAAO,QAAA,CAAS,aAAA,CAAc,MAAM,CAAA;AAC1C,QAAA,IAAA,CAAK,SAAA,GAAY,UAAA;AACjB,QAAA,IAAA,CAAK,eAAA,GAAkB,OAAA;AACvB,QAAA,IAAA,CAAK,QAAQ,IAAA,GAAO,MAAA;AACpB,QAAA,IAAA,CAAK,OAAA,CAAQ,GAAA,GAAM,MAAA,CAAO,IAAA,CAAK,KAAK,CAAA;AACpC,QAAA,IAAA,CAAK,QAAQ,GAAA,GAAM,GAAA;AACnB,QAAA,EAAA,CAAG,YAAY,IAAI,CAAA;AACnB,QAAA,MAAM,WAAW,IAAA,CAAK,IAAA,CAAK,UAAA,CAAW,IAAA,CAAK,KAAK,IAAI,CAAA;AACpD,QAAA,IAAI,QAAA,EAAU;AACZ,UAAA,MAAM,CAAA,GAAIA,kBAAW,IAAI,CAAA;AACzB,UAAA,CAAA,CAAE,MAAA,CAAO,QAAA,CAAS,EAAE,MAAA,EAAQ,IAAA,CAAK,QAAQ,IAAA,EAAM,IAAA,CAAK,IAAA,EAAM,CAAc,CAAA;AACxE,UAAA,IAAA,CAAK,KAAA,CAAM,GAAA,CAAI,IAAA,EAAM,CAAC,CAAA;AAAA,QACxB;AAAA,MACF,CAAA,MAAO;AACL,QAAA,EAAA,CAAG,WAAA,CAAY,QAAA,CAAS,IAAI,CAAC,CAAA;AAAA,MAC/B;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,eAAe,EAAA,EAAiB;AACtC,IAAA,KAAA,MAAW,CAAC,IAAA,EAAM,IAAI,CAAA,IAAK,KAAK,KAAA,EAAO;AACrC,MAAA,IAAI,EAAA,KAAO,IAAA,IAAQ,EAAA,CAAG,QAAA,CAAS,IAAI,CAAA,EAAG;AACpC,QAAA,IAAA,CAAK,OAAA,EAAQ;AACb,QAAA,IAAA,CAAK,KAAA,CAAM,OAAO,IAAI,CAAA;AAAA,MACxB;AAAA,IACF;AAAA,EACF;AAAA;AAAA,EAIQ,aAAA,GAAgB;AACtB,IAAA,MAAM,CAAA,GAAI,OAAO,YAAA,EAAa;AAC9B,IAAA,IAAI,CAAC,CAAA,IAAK,CAAA,CAAE,UAAA,KAAe,CAAA,IAAK,CAAC,IAAA,CAAK,IAAA,CAAK,QAAA,CAAS,CAAA,CAAE,UAAU,CAAA,EAAG,OAAO,IAAA;AAC1E,IAAA,MAAM,IAAI,UAAA,CAAW,IAAA,CAAK,MAAM,CAAA,CAAE,UAAA,EAAY,EAAE,YAAY,CAAA;AAC5D,IAAA,MAAM,IAAI,UAAA,CAAW,IAAA,CAAK,MAAM,CAAA,CAAE,SAAA,EAAW,EAAE,WAAW,CAAA;AAC1D,IAAA,IAAI,CAAC,CAAA,IAAK,CAAC,CAAA,EAAG,OAAO,IAAA;AACrB,IAAA,OAAO,EAAE,MAAA,EAAQ,EAAE,SAAS,CAAA,CAAE,OAAA,EAAS,QAAQ,CAAA,CAAE,MAAA,EAAO,EAAG,KAAA,EAAO,EAAE,OAAA,EAAS,CAAA,CAAE,SAAS,MAAA,EAAQ,CAAA,CAAE,QAAO,EAAE;AAAA,EAC7G;AAAA;AAAA,EAGQ,cAAA,GAAiB;AACvB,IAAA,MAAM,GAAA,GAAM,IAAA,CAAK,MAAA,CAAO,YAAA,EAAa;AACrC,IAAA,IAAI,CAAC,GAAA,EAAK;AACV,IAAA,MAAM,CAAA,GAAI,WAAW,IAAA,CAAK,IAAA,EAAM,IAAI,MAAA,CAAO,OAAA,EAAS,GAAA,CAAI,MAAA,CAAO,MAAM,CAAA;AACrE,IAAA,MAAM,CAAA,GAAI,WAAW,IAAA,CAAK,IAAA,EAAM,IAAI,KAAA,CAAM,OAAA,EAAS,GAAA,CAAI,KAAA,CAAM,MAAM,CAAA;AACnE,IAAA,IAAI,CAAC,CAAA,IAAK,CAAC,CAAA,EAAG;AACd,IAAA,MAAM,CAAA,GAAI,SAAS,WAAA,EAAY;AAC/B,IAAA,MAAM,CAAA,GAAI,OAAO,YAAA,EAAa;AAC9B,IAAA,IAAI,CAAC,CAAA,EAAG;AACR,IAAA,IAAA,CAAK,aAAA,GAAgB,IAAA;AACrB,IAAA,IAAI;AACF,MAAA,CAAA,CAAE,QAAA,CAAS,CAAA,CAAE,IAAA,EAAM,CAAA,CAAE,MAAM,CAAA;AAC3B,MAAA,CAAA,CAAE,eAAA,EAAgB;AAClB,MAAA,CAAA,CAAE,SAAS,CAAC,CAAA;AACZ,MAAA,CAAA,CAAE,MAAA,CAAO,CAAA,CAAE,IAAA,EAAM,CAAA,CAAE,MAAM,CAAA;AAAA,IAC3B,CAAA,CAAA,MAAQ;AAAA,IAER,CAAA,SAAE;AACA,MAAA,IAAA,CAAK,aAAA,GAAgB,KAAA;AAAA,IACvB;AAAA,EACF;AAAA;AAAA,EAGQ,aAAa,EAAA,EAAyB;AAC5C,IAAA,IAAI,GAAA,GAAM,EAAA;AACV,IAAA,KAAA,MAAW,KAAA,IAAS,KAAA,CAAM,IAAA,CAAK,EAAA,CAAG,UAAU,CAAA,EAAG;AAC7C,MAAA,IAAI,KAAA,YAAiB,WAAA,IAAe,KAAA,CAAM,OAAA,CAAQ,QAAQ,IAAA,EAAM;AAC9D,QAAA,GAAA,IAAO,WAAA;AAAA,MACT,CAAA,MAAO;AACL,QAAA,GAAA,IAAO,MAAM,WAAA,IAAe,EAAA;AAAA,MAC9B;AAAA,IACF;AACA,IAAA,OAAO,GAAA;AAAA,EACT;AAAA;AAAA;AAAA,EAKQ,UAAU,CAAA,EAAkB;AAClC,IAAA,IAAI,IAAA,CAAK,KAAK,QAAA,EAAU;AACxB,IAAA,MAAM,GAAA,GAAM,CAAA,CAAE,OAAA,IAAW,CAAA,CAAE,OAAA;AAC3B,IAAA,IAAI,CAAC,GAAA,IAAO,CAAA,CAAE,MAAA,EAAQ;AACtB,IAAA,MAAM,CAAA,GAAI,CAAA,CAAE,GAAA,CAAI,WAAA,EAAY;AAC5B,IAAA,MAAM,IAAA,GAAQ,EAAE,CAAA,EAAG,MAAA,EAAQ,CAAA,EAAG,QAAA,EAAU,CAAA,EAAG,WAAA,EAAa,CAAA,EAAG,MAAA,EAAO,CAAY,CAAC,CAAA;AAC/E,IAAA,IAAI,IAAA,EAAM;AACR,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,MAAM,GAAA,GAAM,KAAK,aAAA,EAAc;AAC/B,MAAA,IAAI,GAAA,EAAK,IAAA,CAAK,MAAA,CAAO,YAAA,CAAa,GAAG,CAAA;AACrC,MAAA,IAAA,CAAK,MAAA,CAAO,WAAW,IAAI,CAAA;AAC3B,MAAA,IAAA,CAAK,MAAA,EAAO;AAAA,IACd,CAAA,MAAA,IAAW,MAAM,GAAA,EAAK;AACpB,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,IAAI,CAAA,CAAE,QAAA,EAAU,IAAA,CAAK,MAAA,CAAO,IAAA,EAAK;AAAA,WAC5B,IAAA,CAAK,OAAO,IAAA,EAAK;AACtB,MAAA,IAAA,CAAK,MAAA,EAAO;AAAA,IACd,CAAA,MAAA,IAAW,MAAM,GAAA,EAAK;AACpB,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,IAAA,CAAK,OAAO,IAAA,EAAK;AACjB,MAAA,IAAA,CAAK,MAAA,EAAO;AAAA,IACd;AAAA,EACF;AAAA,EAEQ,cAAc,CAAA,EAAe;AACnC,IAAA,IAAI,IAAA,CAAK,KAAK,QAAA,EAAU;AACtB,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA;AAAA,IACF;AACA,IAAA,MAAM,GAAA,GAAM,KAAK,aAAA,EAAc;AAC/B,IAAA,IAAI,CAAC,GAAA,EAAK;AACV,IAAA,IAAA,CAAK,MAAA,CAAO,aAAa,GAAG,CAAA;AAC5B,IAAA,MAAM,SAAA,GAAYC,oBAAY,GAAG,CAAA;AACjC,IAAA,MAAM,WAAA,GAAc,KAAK,MAAA,CAAO,gBAAA,IAAoB,KAAA,CAAM,MAAA,IAAU,IAAI,KAAA,CAAM,MAAA;AAC9E,IAAA,MAAM,IAAI,CAAA,CAAE,SAAA;AAIZ,IAAA,IAAI,cAAc,CAAA,KAAM,YAAA,IAAgB,CAAA,KAAM,uBAAA,IAA2B,MAAM,uBAAA,CAAA,EAA0B;AACzG,IAAA,IAAI,SAAA,IAAa,MAAM,sBAAA,EAAwB;AAC/C,IAAA,IAAI,SAAA,IAAa,CAAA,KAAM,uBAAA,IAA2B,WAAA,GAAc,CAAA,EAAG;AAGnE,IAAA,MAAM,KAAK,IAAA,CAAK,MAAA;AAChB,IAAA,IAAI,MAAM,iBAAA,EAAmB;AAC3B,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,EAAA,CAAG,oBAAA,EAAqB;AAAA,IAC1B,CAAA,MAAA,IAAW,CAAA,CAAE,UAAA,CAAW,QAAQ,CAAA,EAAG;AACjC,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,IAAI,CAAA,KAAM,sBAAA,EAAwB,EAAA,CAAG,aAAA,EAAc;AAAA,cAC3C,cAAA,EAAe;AAAA,IACzB,WAAW,CAAA,KAAM,YAAA,IAAgB,CAAA,KAAM,uBAAA,IAA2B,MAAM,iBAAA,EAAmB;AACzF,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,MAAM,OAAO,CAAA,CAAE,IAAA,IAAQ,EAAE,YAAA,EAAc,OAAA,CAAQ,YAAY,CAAA,IAAK,EAAA;AAChE,MAAA,IAAI,IAAA,EAAM,EAAA,CAAG,UAAA,CAAW,IAAI,CAAA;AAAA,IAC9B,CAAA,MAAA,IAAW,MAAM,iBAAA,EAAmB;AAClC,MAAA,CAAA,CAAE,cAAA,EAAe;AACjB,MAAA,EAAA,CAAG,WAAW,IAAI,CAAA;AAAA,IACpB,CAAA,MAAO;AACL,MAAA;AAAA,IACF;AACE,IAAA,IAAA,CAAK,MAAA,EAAO;AAAA,EAChB;AAAA,EAEQ,OAAA,GAAU;AAChB,IAAA,IAAI,IAAA,CAAK,SAAA,IAAa,IAAA,CAAK,IAAA,CAAK,QAAA,EAAU;AAC1C,IAAA,MAAM,OAAA,GAAU,UAAU,MAAA,CAAO,YAAA,IAAgB,UAAA,IAAc,IAAA,EAAM,KAAK,IAAI,CAAA;AAC9E,IAAA,IAAI,CAAC,OAAA,EAAS;AAEZ,MAAA,IAAA,CAAK,YAAA,EAAa;AAClB,MAAA,IAAA,CAAK,YAAA,GAAe,KAAK,GAAA,EAAI;AAC7B,MAAA;AAAA,IACF;AACA,IAAA,MAAM,EAAA,GAAK,QAAQ,OAAA,CAAQ,OAAA;AAC3B,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,YAAA,CAAa,OAAO,CAAA;AACtC,IAAA,MAAM,GAAA,GAAM,IAAA,CAAK,MAAA,CAAO,YAAA,CAAa,EAAE,CAAA;AACvC,IAAA,IAAI,SAAS,GAAA,EAAK;AAGlB,IAAA,MAAM,MAAM,IAAA,CAAK,GAAA,CAAI,GAAA,CAAI,MAAA,EAAQ,KAAK,MAAM,CAAA;AAC5C,IAAA,IAAI,CAAA,GAAI,CAAA;AACR,IAAA,OAAO,IAAI,GAAA,IAAO,GAAA,CAAI,CAAC,CAAA,KAAM,IAAA,CAAK,CAAC,CAAA,EAAG,CAAA,EAAA;AACtC,IAAA,IAAI,CAAA,GAAI,CAAA;AACR,IAAA,OAAO,CAAA,GAAI,GAAA,GAAM,CAAA,IAAK,GAAA,CAAI,IAAI,MAAA,GAAS,CAAA,GAAI,CAAC,CAAA,KAAM,IAAA,CAAK,IAAA,CAAK,MAAA,GAAS,CAAA,GAAI,CAAC,CAAA,EAAG,CAAA,EAAA;AAC7E,IAAA,MAAM,IAAA,GAAO,CAAA;AACb,IAAA,MAAM,EAAA,GAAK,IAAI,MAAA,GAAS,CAAA;AACxB,IAAA,MAAM,SAAS,IAAA,CAAK,KAAA,CAAM,CAAA,EAAG,IAAA,CAAK,SAAS,CAAC,CAAA;AAC5C,IAAA,IAAA,CAAK,OAAO,YAAA,CAAa,EAAE,MAAA,EAAQ,EAAE,SAAS,EAAA,EAAI,MAAA,EAAQ,IAAA,EAAK,EAAG,OAAO,EAAE,OAAA,EAAS,IAAI,MAAA,EAAQ,EAAA,IAAM,CAAA;AACtG,IAAA,IAAI,EAAA,GAAK,IAAA,EAAM,IAAA,CAAK,MAAA,CAAO,cAAA,EAAe;AAC1C,IAAA,IAAI,MAAA,EAAQ,IAAA,CAAK,MAAA,CAAO,UAAA,CAAW,MAAM,CAAA;AAEzC,IAAA,IAAA,CAAK,QAAQ,OAAO,CAAA;AACpB,IAAA,IAAA,CAAK,YAAA,GAAe,KAAK,GAAA,EAAI;AAAA,EAC/B;AAAA;AAAA,EAGQ,QAAQ,EAAA,EAAiB;AAC/B,IAAA,IAAI,GAAA,GAAM,CAAA;AACV,IAAA,KAAA,MAAW,KAAA,IAAS,KAAA,CAAM,IAAA,CAAK,EAAA,CAAG,QAAQ,CAAA,EAAoB;AAC5D,MAAA,IAAI,KAAA,CAAM,OAAA,CAAQ,GAAA,IAAO,IAAA,EAAM;AAC/B,MAAA,KAAA,CAAM,OAAA,CAAQ,GAAA,GAAM,MAAA,CAAO,GAAG,CAAA;AAC9B,MAAA,MAAM,GAAA,GAAM,MAAM,OAAA,CAAQ,IAAA,IAAQ,OAAO,CAAA,GAAA,CAAK,KAAA,CAAM,eAAe,EAAA,EAAI,MAAA;AACvE,MAAA,KAAA,CAAM,OAAA,CAAQ,GAAA,GAAM,MAAA,CAAO,GAAG,CAAA;AAC9B,MAAA,GAAA,IAAO,GAAA;AAAA,IACT;AAAA,EACF;AACF,CAAA;ACnWA,SAAS,eAAA,GAAkC;AACzC,EAAA,MAAM,CAAA,GAAI,OAAO,YAAA,EAAa;AAC9B,EAAA,IAAI,CAAC,CAAA,IAAK,CAAA,CAAE,UAAA,KAAe,GAAG,OAAO,IAAA;AACrC,EAAA,MAAM,CAAA,GAAI,CAAA,CAAE,UAAA,CAAW,CAAC,EAAE,UAAA,EAAW;AACrC,EAAA,CAAA,CAAE,QAAA,CAAS,CAAA,CAAE,SAAA,KAAc,CAAA,CAAE,YAAA,IAAgB,EAAE,WAAA,KAAgB,CAAA,CAAE,SAAA,GAAY,KAAA,GAAQ,IAAI,CAAA;AACzF,EAAA,MAAM,KAAA,GAAQ,EAAE,cAAA,EAAe;AAC/B,EAAA,IAAI,MAAM,MAAA,EAAQ,OAAO,KAAA,CAAM,KAAA,CAAM,SAAS,CAAC,CAAA;AAC/C,EAAA,MAAM,CAAA,GAAI,EAAE,qBAAA,EAAsB;AAClC,EAAA,OAAO,CAAA,CAAE,MAAA,IAAU,CAAA,CAAE,KAAA,GAAQ,CAAA,GAAI,IAAA;AACnC;AAQO,IAAM,aAAaC,gBAAA,CAA8C,SAASC,WAAAA,CAC/E,EAAE,QAAQ,SAAA,EAAW,KAAA,EAAO,QAAA,GAAW,GAAA,EAAK,aAAa,SAAA,EAAW,QAAA,EAAU,cAAA,EAAgB,aAAA,IAC9F,GAAA,EACA;AACA,EAAA,MAAM,QAAA,GAAW,kBAAkB,MAAM,CAAA;AACzC,EAAA,MAAM,WAAA,GAAcT,aAAuB,IAAI,CAAA;AAC/C,EAAA,MAAM,UAAA,GAAaA,aAAuB,IAAI,CAAA;AAC9C,EAAA,MAAM,OAAA,GAAUA,aAA0B,IAAI,CAAA;AAC9C,EAAA,MAAM,CAAC,OAAA,EAAS,UAAU,CAAA,GAAIU,eAAS,KAAK,CAAA;AAC5C,EAAA,MAAM,CAAC,KAAA,EAAO,QAAQ,CAAA,GAAIA,eAAqD,IAAI,CAAA;AAGnF,EAAA,MAAM,YAAA,GAAeV,YAAAA,CAAO,EAAE,cAAA,EAAgB,eAAe,CAAA;AAC7D,EAAA,YAAA,CAAa,OAAA,GAAU,EAAE,cAAA,EAAgB,aAAA,EAAc;AAEvD,EAAAW,yBAAA;AAAA,IACE,GAAA;AAAA,IACA,OAAyB;AAAA,MACvB,KAAA,EAAO,MAAM,UAAA,CAAW,OAAA,EAAS,KAAA,EAAM;AAAA,MACvC,cAAc,MAAM;AAClB,QAAA,MAAM,IAAI,eAAA,EAAgB;AAC1B,QAAA,OAAO,CAAA,GAAI,EAAE,CAAA,EAAG,CAAA,CAAE,IAAA,EAAM,CAAA,EAAG,CAAA,CAAE,GAAA,EAAK,MAAA,EAAQ,CAAA,CAAE,MAAA,IAAU,EAAA,EAAG,GAAI,IAAA;AAAA,MAC/D,CAAA;AAAA,MACA,kBAAkB,MAAM;AACtB,QAAA,MAAM,CAAA,GAAI,OAAO,YAAA,EAAa;AAC9B,QAAA,IAAI,CAAC,CAAA,IAAK,CAAA,CAAE,eAAe,CAAA,IAAK,CAAA,CAAE,aAAa,OAAO,IAAA;AACtD,QAAA,MAAM,CAAA,GAAI,CAAA,CAAE,UAAA,CAAW,CAAC,EAAE,qBAAA,EAAsB;AAChD,QAAA,IAAI,CAAC,CAAA,CAAE,KAAA,IAAS,CAAC,CAAA,CAAE,QAAQ,OAAO,IAAA;AAClC,QAAA,OAAO,EAAE,GAAA,EAAK,CAAA,CAAE,KAAK,IAAA,EAAM,CAAA,CAAE,MAAM,KAAA,EAAO,CAAA,CAAE,KAAA,EAAO,MAAA,EAAQ,EAAE,MAAA,EAAQ,KAAA,EAAO,EAAE,KAAA,EAAO,MAAA,EAAQ,EAAE,MAAA,EAAO;AAAA,MACxG,CAAA;AAAA,MACA,gBAAA,EAAkB,MAAM,WAAA,CAAY;AAAA,KACtC,CAAA;AAAA,IACA;AAAC,GACH;AAGA,EAAAR,gBAAU,MAAM;AACd,IAAA,MAAM,KAAK,UAAA,CAAW,OAAA;AACtB,IAAA,IAAI,CAAC,EAAA,EAAI;AACT,IAAA,MAAM,IAAA,GAAO,IAAI,UAAA,CAAW,EAAA,EAAI,MAAA,EAAQ;AAAA,MACtC,QAAA;AAAA,MACA,YAAY,CAAC,CAAA,KAAM,YAAA,CAAa,OAAA,CAAQ,gBAAgB,CAAC,CAAA;AAAA,MACzD,aAAa,CAAC,CAAA,KAAM,YAAA,CAAa,OAAA,CAAQ,iBAAiB,CAAC;AAAA,KAC5D,CAAA;AACD,IAAA,OAAA,CAAQ,OAAA,GAAU,IAAA;AAClB,IAAA,OAAO,MAAM;AACX,MAAA,IAAA,CAAK,OAAA,EAAQ;AACb,MAAA,OAAA,CAAQ,OAAA,GAAU,IAAA;AAAA,IACpB,CAAA;AAAA,EACF,CAAA,EAAG,CAAC,MAAA,EAAQ,QAAQ,CAAC,CAAA;AAIrB,EAAAA,gBAAU,MAAM;AACd,IAAA,OAAA,CAAQ,SAAS,IAAA,EAAK;AAAA,EACxB,CAAA,EAAG,CAAC,QAAA,CAAS,QAAQ,CAAC,CAAA;AAEtB,EAAAA,gBAAU,MAAM;AACd,IAAA,IAAI,SAAA,EAAW,UAAA,CAAW,OAAA,EAAS,KAAA,EAAM;AAAA,EAC3C,CAAA,EAAG,CAAC,SAAS,CAAC,CAAA;AAGd,EAAAA,gBAAU,MAAM;AACd,IAAA,MAAM,SAAS,MAAM;AACnB,MAAA,MAAM,UAAU,UAAA,CAAW,OAAA;AAC3B,MAAA,MAAM,CAAA,GAAI,OAAO,YAAA,EAAa;AAC9B,MAAA,IAAI,CAAC,OAAA,IAAW,CAAC,CAAA,IAAK,EAAE,UAAA,KAAe,CAAA,IAAK,CAAC,CAAA,CAAE,eAAe,CAAC,OAAA,CAAQ,QAAA,CAAS,CAAA,CAAE,UAAU,CAAA,EAAG;AAC7F,QAAA,QAAA,CAAS,IAAI,CAAA;AACb,QAAA;AAAA,MACF;AACA,MAAA,MAAM,IAAI,eAAA,EAAgB;AAC1B,MAAA,MAAM,GAAA,GAAM,QAAQ,qBAAA,EAAsB;AAC1C,MAAA,IAAI,GAAG,QAAA,CAAS,EAAE,CAAA,EAAG,CAAA,CAAE,OAAO,GAAA,CAAI,IAAA,EAAM,CAAA,EAAG,CAAA,CAAE,MAAM,GAAA,CAAI,GAAA,EAAK,GAAG,CAAA,CAAE,MAAA,IAAU,IAAI,CAAA;AAAA,IACjF,CAAA;AACA,IAAA,QAAA,CAAS,gBAAA,CAAiB,mBAAmB,MAAM,CAAA;AACnD,IAAA,MAAM,EAAA,GAAK,IAAI,cAAA,CAAe,MAAM,CAAA;AACpC,IAAA,IAAI,UAAA,CAAW,OAAA,EAAS,EAAA,CAAG,OAAA,CAAQ,WAAW,OAAO,CAAA;AACrD,IAAA,MAAA,EAAO;AACP,IAAA,OAAO,MAAM;AACX,MAAA,QAAA,CAAS,mBAAA,CAAoB,mBAAmB,MAAM,CAAA;AACtD,MAAA,EAAA,CAAG,UAAA,EAAW;AAAA,IAChB,CAAA;AAAA,EACF,CAAA,EAAG,EAAE,CAAA;AAGL,EAAAS,qBAAA,CAAgB,MAAM;AACpB,IAAA,MAAM,KAAK,WAAA,CAAY,OAAA;AACvB,IAAA,MAAM,UAAU,UAAA,CAAW,OAAA;AAC3B,IAAA,IAAI,CAAC,EAAA,IAAM,CAAC,OAAA,EAAS;AACrB,IAAA,MAAM,OAAO,MAAM;AACjB,MAAA,MAAA,CAAO,QAAA,CAAS,QAAQ,WAAW,CAAA;AACnC,MAAA,MAAA,CAAO,WAAA,CAAY,EAAA,CAAG,SAAA,EAAW,EAAA,CAAG,YAAY,CAAA;AAAA,IAClD,CAAA;AACA,IAAA,IAAA,EAAK;AACL,IAAA,MAAM,EAAA,GAAK,IAAI,cAAA,CAAe,IAAI,CAAA;AAClC,IAAA,EAAA,CAAG,QAAQ,EAAE,CAAA;AACb,IAAA,EAAA,CAAG,QAAQ,OAAO,CAAA;AAClB,IAAA,OAAO,MAAM,GAAG,UAAA,EAAW;AAAA,EAC7B,CAAA,EAAG,CAAC,MAAM,CAAC,CAAA;AAEX,EAAA,MAAM,WAAW,MAAM;AACrB,IAAA,MAAM,KAAK,WAAA,CAAY,OAAA;AACvB,IAAA,IAAI,IAAI,MAAA,CAAO,WAAA,CAAY,EAAA,CAAG,SAAA,EAAW,GAAG,YAAY,CAAA;AAAA,EAC1D,CAAA;AAEA,EAAA,MAAM,SAAA,GAAY,OAAA,IAAW,CAAC,CAAC,SAAS,CAAC,QAAA;AAEzC,EAAA,uBACEC,cAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAW,CAAA,QAAA,EAAW,SAAA,GAAY,CAAA,CAAA,EAAI,SAAS,CAAA,CAAA,GAAK,EAAE,CAAA,CAAA,EAAI,KAAA,EAC7D,QAAA,kBAAAA,cAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,cAAA,EAAe,GAAA,EAAK,WAAA,EAAa,QAAA,EAC9C,QAAA,kBAAAC,eAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,aAAA,EAAc,KAAA,EAAO,EAAE,QAAA,EAAU,YAAA,EAAc,MAAA,EAAQ,QAAA,EAAU,YAAW,EACzF,QAAA,EAAA;AAAA,oBAAAD,cAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,SAAA,EAAU,mBAAA;AAAA,QACV,GAAA,EAAK,UAAA;AAAA,QACL,OAAA,EAAS,MAAM,UAAA,CAAW,IAAI,CAAA;AAAA,QAC9B,MAAA,EAAQ,MAAM,UAAA,CAAW,KAAK,CAAA;AAAA,QAC9B,8BAAA,EAA8B;AAAA;AAAA,KAChC;AAAA,IACC,aAAa,KAAA,mBACZA,cAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,SAAA,EAAU,WAAA;AAAA,QACV,KAAA,EAAO,EAAE,QAAA,EAAU,UAAA,EAAY,MAAM,KAAA,CAAM,CAAA,EAAG,GAAA,EAAK,KAAA,CAAM,CAAA,EAAG,MAAA,EAAQ,KAAA,CAAM,CAAA,EAAG,eAAe,MAAA,EAAO;AAAA,QACnG,aAAA,EAAW;AAAA;AAAA,KACb,GACE,IAAA;AAAA,IACH,QAAA,CAAS,KAAA,IAAS,WAAA,mBACjBA,cAAA,CAAC,KAAA,EAAA,EAAI,WAAU,iBAAA,EAAkB,aAAA,EAAW,IAAA,EACzC,QAAA,EAAA,WAAA,EACH,CAAA,GACE;AAAA,GAAA,EACN,GACF,CAAA,EACF,CAAA;AAEJ,CAAC;ACjLD,IAAM,QAAmB,EAAE,MAAA,EAAQ,EAAC,EAAG,KAAA,EAAO,EAAC,EAAE;AAEjD,IAAM,gBAAA,GAAmBE,oBAAyB,KAAK,CAAA;AACtB,gBAAA,CAAiB;AAC3C,IAAM,YAAA,GAAe,MAAiBC,gBAAA,CAAW,gBAAgB","file":"index.cjs","sourcesContent":["import {\n EditorController,\n createCanvasMeasurer,\n type EditorSchema,\n type Measurer,\n type Typography,\n} from \"@wingleeio/ori-core\";\nimport { useEffect, useRef } from \"react\";\nimport type * as Y from \"yjs\";\n\nexport interface UseEditorOptions {\n /** Existing note `Y.Doc`. When switching notes, remount via `key` instead. */\n doc?: Y.Doc;\n typography?: Typography;\n measurer?: Measurer;\n overscan?: number;\n blockSpacing?: number;\n /** Custom block/atom nodes, merged over the built-ins. */\n schema?: Partial<EditorSchema>;\n}\n\n/**\n * Create (once) and own an {@link EditorController} for the lifetime of the\n * component. To switch documents, give the hosting component a `key` so it\n * remounts with a fresh controller.\n *\n * The controller is reconnected on mount and disconnected on unmount rather than\n * destroyed, so React StrictMode's dev mount → unmount → remount cycle reuses the\n * same controller (with all its state) instead of leaving a torn-down one behind.\n */\nexport function useEditor(options: UseEditorOptions = {}): EditorController {\n const ref = useRef<EditorController | null>(null);\n if (ref.current === null) {\n ref.current = new EditorController({\n doc: options.doc,\n measurer: options.measurer ?? createCanvasMeasurer(),\n typography: options.typography,\n overscan: options.overscan,\n blockSpacing: options.blockSpacing,\n schema: options.schema,\n });\n }\n useEffect(() => {\n const editor = ref.current;\n editor?.connect();\n return () => editor?.disconnect();\n }, []);\n return ref.current;\n}\n","import type { EditorController, EditorSnapshot, Marks } from \"@wingleeio/ori-core\";\nimport { useSyncExternalStore } from \"react\";\n\n/** Subscribe a component to the controller's snapshot stream. */\nexport function useEditorSnapshot(editor: EditorController): EditorSnapshot {\n return useSyncExternalStore(editor.subscribe, editor.getSnapshot, editor.getSnapshot);\n}\n\n/**\n * The marks active at the current selection. Recomputed whenever the editor\n * notifies (selection move, edit, or pending-mark toggle), so toolbars stay in\n * sync without their own subscription.\n */\nexport function useActiveMarks(editor: EditorController): Marks {\n const snapshot = useEditorSnapshot(editor);\n // `revision` changes on every notify; reading it ties this to the store.\n void snapshot.revision;\n return editor.getActiveMarks();\n}\n","import type { InlineItem } from \"@wingleeio/ori-core\";\n\n/** CSS.escape with a fallback (jsdom lacks it). */\nexport function esc(s: string): string {\n return typeof CSS !== \"undefined\" && CSS.escape ? CSS.escape(s) : s.replace(/[\"\\\\]/g, \"\\\\$&\");\n}\n\n/**\n * Imperative DOM helpers for the contentEditable view. Each block is a\n * block-level element carrying `data-block-id`; its inline runs are spans\n * carrying `data-off` (their start offset in the block) so DOM positions map\n * back to {blockId, offset} and vice-versa.\n */\n\nexport const BLOCK_SEL = \"[data-block-id]\";\n\nexport function blockElOf(node: Node | null, root: HTMLElement): HTMLElement | null {\n let n: Node | null = node;\n while (n && n !== root) {\n if (n instanceof HTMLElement && n.dataset.blockId) return n;\n n = n.parentNode;\n }\n return null;\n}\n\nfunction spanOf(node: Node | null): HTMLElement | null {\n let n: Node | null = node;\n while (n) {\n if (n instanceof HTMLElement && n.dataset.off != null) return n;\n n = n.parentNode;\n }\n return null;\n}\n\n/** Map a DOM (node, offset) to a {blockId, offset} model position. */\nexport function domToModel(\n root: HTMLElement,\n node: Node | null,\n offset: number,\n): { blockId: string; offset: number } | null {\n const blockEl = blockElOf(node, root);\n if (!blockEl) return null;\n const blockId = blockEl.dataset.blockId as string;\n\n if (node && node.nodeType === Node.TEXT_NODE) {\n const span = spanOf(node);\n const base = span ? Number(span.dataset.off) : 0;\n return { blockId, offset: base + offset };\n }\n\n // node is an element; `offset` is a child index. Resolve via the child spans.\n const el = node as HTMLElement;\n if (el.dataset?.off != null) {\n // selection landed on a span boundary\n return { blockId, offset: Number(el.dataset.off) + (offset > 0 ? spanLen(el) : 0) };\n }\n const kids = Array.from(el.childNodes);\n for (let i = offset; i < kids.length; i++) {\n const k = kids[i];\n if (k instanceof HTMLElement && k.dataset.off != null) return { blockId, offset: Number(k.dataset.off) };\n }\n // past the last span → block end\n let end = 0;\n for (const k of kids) if (k instanceof HTMLElement && k.dataset.off != null) end = Math.max(end, Number(k.dataset.off) + spanLen(k));\n return { blockId, offset: end };\n}\n\nfunction spanLen(span: HTMLElement): number {\n return span.dataset.len != null ? Number(span.dataset.len) : (span.textContent ?? \"\").length;\n}\n\n/** Find the DOM (node, offset) for a {blockId, offset} model position. */\nexport function modelToDom(\n root: HTMLElement,\n blockId: string,\n offset: number,\n): { node: Node; offset: number } | null {\n const blockEl = root.querySelector(`[data-block-id=\"${esc(blockId)}\"]`) as HTMLElement | null;\n if (!blockEl) return null;\n const spans = Array.from(blockEl.querySelectorAll(\"[data-off]\")) as HTMLElement[];\n if (spans.length === 0) {\n return { node: blockEl, offset: 0 }; // empty block\n }\n for (const span of spans) {\n const start = Number(span.dataset.off);\n const len = spanLen(span);\n if (offset <= start + len) {\n if (span.dataset.atom != null) {\n // atom: place before or after it (offset is start or start+1)\n const idx = Array.prototype.indexOf.call(blockEl.childNodes, span);\n return { node: blockEl, offset: offset <= start ? idx : idx + 1 };\n }\n const textNode = span.firstChild ?? span;\n return { node: textNode, offset: Math.max(0, Math.min(offset - start, (textNode.textContent ?? \"\").length)) };\n }\n }\n // past everything → after the last span\n const last = spans[spans.length - 1];\n const textNode = last.firstChild ?? last;\n return { node: textNode, offset: (textNode.textContent ?? \"\").length };\n}\n\nfunction markClass(marks: InlineItem[\"marks\"]): string {\n const m = marks ?? {};\n const cls = [\"ori-frag\"];\n if (m.bold) cls.push(\"ori-m-bold\");\n if (m.italic) cls.push(\"ori-m-italic\");\n if (m.underline) cls.push(\"ori-m-underline\");\n if (m.strike) cls.push(\"ori-m-strike\");\n if (m.code) cls.push(\"ori-frag-code\");\n if (m.link) cls.push(\"ori-frag-link\");\n return cls.join(\" \");\n}\n\n/** Build the inline run DOM for a block (text spans only; atoms handled by the view). */\nexport function buildRun(item: InlineItem): HTMLElement {\n const span = document.createElement(\"span\");\n span.className = markClass(item.marks);\n span.dataset.off = String(item.start);\n span.dataset.len = String(item.text.length);\n span.textContent = item.text;\n return span;\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport { isCollapsed } from \"@wingleeio/ori-core\";\nimport type { ReactNode } from \"react\";\nimport { createRoot, type Root } from \"react-dom/client\";\nimport type { AtomRenderer, BlockRenderer } from \"../renderers\";\nimport { blockElOf, buildRun, domToModel, esc, modelToDom } from \"./dom\";\n\nconst PLACEHOLDER = \"\";\n\nexport interface ViewOptions {\n readOnly?: boolean;\n renderAtom: (type: string) => AtomRenderer | undefined;\n renderBlock: (type: string) => BlockRenderer | undefined;\n}\n\n/**\n * Imperative contentEditable view over an {@link EditorController}. The browser\n * owns caret / selection / trackpad / menus / IME on the live text; we intercept\n * structural + cross-block edits (beforeinput) and route them through the\n * controller, let smooth in-block typing flow natively and read it back, and\n * keep the DOM selection and the controller selection in lock-step.\n */\nexport class EditorView {\n private roots = new Map<HTMLElement, Root>();\n private composing = false;\n private applyingModel = false;\n private detachers: Array<() => void> = [];\n /** The model revision the DOM currently reflects (so external changes — remote\n * edits, app commands — re-render, but our own edits don't clobber the caret). */\n private lastRevision = -1;\n\n constructor(\n private root: HTMLElement,\n private editor: EditorController,\n private opts: ViewOptions,\n ) {\n root.setAttribute(\"contenteditable\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"spellcheck\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"role\", \"textbox\");\n root.setAttribute(\"aria-multiline\", \"true\");\n this.renderBlocks();\n this.lastRevision = this.rev();\n\n const on = <K extends keyof HTMLElementEventMap>(\n t: K,\n h: (e: HTMLElementEventMap[K]) => void,\n o?: AddEventListenerOptions,\n ) => {\n root.addEventListener(t, h as EventListener, o);\n this.detachers.push(() => root.removeEventListener(t, h as EventListener, o));\n };\n on(\"beforeinput\", (e) => this.onBeforeInput(e as InputEvent));\n on(\"input\", () => this.onInput());\n on(\"keydown\", (e) => this.onKeyDown(e as KeyboardEvent));\n on(\"compositionstart\", () => (this.composing = true));\n on(\"compositionend\", () => {\n this.composing = false;\n this.onInput();\n });\n\n const onSelChange = () => {\n if (this.applyingModel || this.composing) return;\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n // DOM is already the source of truth here — record the revision so the\n // resulting React sync() doesn't write the selection back and collapse it.\n this.lastRevision = this.rev();\n };\n document.addEventListener(\"selectionchange\", onSelChange);\n this.detachers.push(() => document.removeEventListener(\"selectionchange\", onSelChange));\n }\n\n destroy() {\n this.detachers.forEach((d) => d());\n this.roots.forEach((r) => r.unmount());\n this.roots.clear();\n }\n\n focus() {\n this.root.focus();\n }\n\n // --- rendering ---------------------------------------------------------\n\n private rev(): number {\n return this.editor.getSnapshot().revision;\n }\n\n /**\n * Called by React on every model change. Only re-renders when the model moved\n * ahead of what we last drew (an *external* change — app command, undo, remote);\n * our own edits already updated the DOM and must not be clobbered.\n */\n sync() {\n const rev = this.rev();\n if (rev === this.lastRevision) return;\n // Only restore the DOM selection when the *content* changed (app command,\n // undo, remote). A selection-only change is already correct in the DOM, and\n // writing it back would fight (and collapse) the user's native selection.\n const changed = this.renderBlocks();\n if (changed) this.writeSelection();\n this.lastRevision = rev;\n }\n\n /** After a controlled (preventDefault'd) edit: re-render + restore the caret. */\n private commit() {\n this.renderBlocks();\n this.writeSelection();\n this.lastRevision = this.rev();\n }\n\n /** A content signature for a block, so unchanged blocks aren't re-rendered. */\n private sig(id: string): string {\n return this.editor.getBlockType(id) + \"|\" + JSON.stringify(this.editor.getInline(id));\n }\n\n /**\n * Reconcile the DOM to the *visible window* of blocks (virtualization): a top\n * spacer, the windowed blocks, then a bottom spacer — heights from the\n * controller's offscreen measurement. On-screen blocks are reused by id so a\n * caret inside one survives a scroll. Returns true if the DOM was mutated.\n */\n private renderBlocks(): boolean {\n let changed = false;\n const snap = this.editor.getSnapshot();\n const vis = snap.visible;\n const topH = vis.length ? vis[0].top : 0;\n const botH = vis.length\n ? Math.max(0, snap.totalHeight - (vis[vis.length - 1].top + vis[vis.length - 1].height))\n : Math.max(0, snap.totalHeight);\n\n const TOP = \"\u0000top\";\n const BOTTOM = \"\u0000bottom\";\n const want = [TOP, ...vis.map((v) => v.id), BOTTOM];\n const keyOf = (el: Element) => {\n const e = el as HTMLElement;\n return e.dataset.spacer ? \"\u0000\" + e.dataset.spacer : (e.dataset.blockId ?? \"\");\n };\n const have = new Map<string, HTMLElement>();\n for (const c of Array.from(this.root.children)) have.set(keyOf(c), c as HTMLElement);\n\n let prev: HTMLElement | null = null;\n for (const k of want) {\n let el: HTMLElement | undefined = have.get(k);\n if (el) {\n have.delete(k);\n } else {\n el = k === TOP || k === BOTTOM ? this.makeSpacer(k.slice(1)) : this.makeBlock(k);\n changed = true;\n }\n const anchor: ChildNode | null = prev ? prev.nextSibling : this.root.firstChild;\n if (anchor !== el) {\n this.root.insertBefore(el, anchor);\n changed = true;\n }\n prev = el;\n }\n for (const el of have.values()) {\n if (el.dataset.blockId) this.unmountRootsIn(el);\n el.remove();\n changed = true;\n }\n\n const top = this.root.firstElementChild as HTMLElement | null;\n if (top && top.style.height !== `${topH}px`) top.style.height = `${topH}px`;\n const bot = this.root.lastElementChild as HTMLElement | null;\n if (bot && bot.style.height !== `${botH}px`) bot.style.height = `${botH}px`;\n\n for (const vb of vis) {\n const el = this.root.querySelector(`[data-block-id=\"${esc(vb.id)}\"]`) as HTMLElement | null;\n if (!el) continue;\n const sig = this.sig(vb.id);\n if (el.dataset.sig !== sig) {\n el.dataset.sig = sig;\n this.renderBlockInner(el, vb.id);\n changed = true;\n }\n }\n return changed;\n }\n\n private makeBlock(id: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.blockId = id;\n return el;\n }\n\n private makeSpacer(which: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.spacer = which;\n el.setAttribute(\"contenteditable\", \"false\");\n el.setAttribute(\"aria-hidden\", \"true\");\n el.style.userSelect = \"none\";\n el.style.pointerEvents = \"none\";\n return el;\n }\n\n private renderBlockInner(el: HTMLElement, id: string) {\n this.unmountRootsIn(el);\n const type = this.editor.getBlockType(id);\n el.className = `ori-block ori-block-${type}`;\n\n const blockRenderer = this.opts.renderBlock(type);\n if (blockRenderer) {\n el.contentEditable = \"false\";\n el.textContent = \"\";\n const root = createRoot(el);\n root.render(blockRenderer({ editor: this.editor, block: { id, type, index: 0, top: 0, height: 0 }, layout: this.editor.getLayout(id)! }) as ReactNode);\n this.roots.set(el, root);\n return;\n }\n el.contentEditable = \"inherit\";\n el.textContent = \"\";\n const items = this.editor.getInline(id);\n if (items.length === 0) {\n el.appendChild(document.createElement(\"br\")); // keep an empty block selectable\n return;\n }\n for (const item of items) {\n if (item.atom) {\n const span = document.createElement(\"span\");\n span.className = \"ori-atom\";\n span.contentEditable = \"false\";\n span.dataset.atom = \"true\";\n span.dataset.off = String(item.start);\n span.dataset.len = \"1\";\n el.appendChild(span);\n const renderer = this.opts.renderAtom(item.atom.type);\n if (renderer) {\n const r = createRoot(span);\n r.render(renderer({ editor: this.editor, atom: item.atom }) as ReactNode);\n this.roots.set(span, r);\n }\n } else {\n el.appendChild(buildRun(item));\n }\n }\n }\n\n private unmountRootsIn(el: HTMLElement) {\n for (const [node, root] of this.roots) {\n if (el === node || el.contains(node)) {\n root.unmount();\n this.roots.delete(node);\n }\n }\n }\n\n // --- selection ---------------------------------------------------------\n\n private readSelection() {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || !this.root.contains(s.anchorNode)) return null;\n const a = domToModel(this.root, s.anchorNode, s.anchorOffset);\n const f = domToModel(this.root, s.focusNode, s.focusOffset);\n if (!a || !f) return null;\n return { anchor: { blockId: a.blockId, offset: a.offset }, focus: { blockId: f.blockId, offset: f.offset } };\n }\n\n /** Push the controller's selection back into the DOM (after a model op). */\n private writeSelection() {\n const sel = this.editor.getSelection();\n if (!sel) return;\n const a = modelToDom(this.root, sel.anchor.blockId, sel.anchor.offset);\n const f = modelToDom(this.root, sel.focus.blockId, sel.focus.offset);\n if (!a || !f) return;\n const r = document.createRange();\n const s = window.getSelection();\n if (!s) return;\n this.applyingModel = true;\n try {\n r.setStart(a.node, a.offset);\n s.removeAllRanges();\n s.addRange(r);\n s.extend(f.node, f.offset);\n } catch {\n /* node detached mid-reconcile */\n } finally {\n this.applyingModel = false;\n }\n }\n\n /** The block text as the model sees it (atoms collapse to one placeholder). */\n private domBlockText(el: HTMLElement): string {\n let out = \"\";\n for (const child of Array.from(el.childNodes)) {\n if (child instanceof HTMLElement && child.dataset.atom != null) {\n out += PLACEHOLDER;\n } else {\n out += child.textContent ?? \"\";\n }\n }\n return out;\n }\n\n // --- input -------------------------------------------------------------\n\n /** Formatting + history shortcuts (the browser fires these as keydown). */\n private onKeyDown(e: KeyboardEvent) {\n if (this.opts.readOnly) return;\n const mod = e.metaKey || e.ctrlKey;\n if (!mod || e.altKey) return;\n const k = e.key.toLowerCase();\n const mark = ({ b: \"bold\", i: \"italic\", u: \"underline\", e: \"code\" } as const)[k];\n if (mark) {\n e.preventDefault();\n const sel = this.readSelection();\n if (sel) this.editor.setSelection(sel);\n this.editor.toggleMark(mark);\n this.commit();\n } else if (k === \"z\") {\n e.preventDefault();\n if (e.shiftKey) this.editor.redo();\n else this.editor.undo();\n this.commit();\n } else if (k === \"y\") {\n e.preventDefault();\n this.editor.redo();\n this.commit();\n }\n }\n\n private onBeforeInput(e: InputEvent) {\n if (this.opts.readOnly) {\n e.preventDefault();\n return;\n }\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n const collapsed = isCollapsed(sel);\n const startOffset = this.editor.orderedSelection()?.start.offset ?? sel.focus.offset;\n const t = e.inputType;\n\n // Native fast path: collapsed in-block typing / deletion. The browser mutates\n // a single text node; onInput reads it back. Keeps autocorrect/IME native.\n if (collapsed && (t === \"insertText\" || t === \"insertCompositionText\" || t === \"insertReplacementText\")) return;\n if (collapsed && t === \"deleteContentForward\") return;\n if (collapsed && t === \"deleteContentBackward\" && startOffset > 0) return;\n\n // Everything else (structural + cross-block) is handled through the controller.\n const ed = this.editor;\n if (t === \"insertParagraph\") {\n e.preventDefault();\n ed.insertParagraphBreak();\n } else if (t.startsWith(\"delete\")) {\n e.preventDefault();\n if (t === \"deleteContentForward\") ed.deleteForward();\n else ed.deleteBackward();\n } else if (t === \"insertText\" || t === \"insertReplacementText\" || t === \"insertFromPaste\") {\n e.preventDefault();\n const text = e.data ?? e.dataTransfer?.getData(\"text/plain\") ?? \"\";\n if (text) ed.insertText(text);\n } else if (t === \"insertLineBreak\") {\n e.preventDefault();\n ed.insertText(\"\\n\");\n } else {\n return; // let the browser handle anything we don't model\n }\n this.commit();\n }\n\n private onInput() {\n if (this.composing || this.opts.readOnly) return;\n const blockEl = blockElOf(window.getSelection()?.anchorNode ?? null, this.root);\n if (!blockEl) {\n // structure changed under us (browser merged blocks) → full resync\n this.renderBlocks();\n this.lastRevision = this.rev();\n return;\n }\n const id = blockEl.dataset.blockId as string;\n const next = this.domBlockText(blockEl);\n const cur = this.editor.getBlockText(id);\n if (next === cur) return;\n\n // diff → splice through the controller (which infers marks at the caret)\n const max = Math.min(cur.length, next.length);\n let p = 0;\n while (p < max && cur[p] === next[p]) p++;\n let s = 0;\n while (s < max - p && cur[cur.length - 1 - s] === next[next.length - 1 - s]) s++;\n const from = p;\n const to = cur.length - s;\n const insert = next.slice(p, next.length - s);\n this.editor.setSelection({ anchor: { blockId: id, offset: from }, focus: { blockId: id, offset: to } });\n if (to > from) this.editor.deleteBackward();\n if (insert) this.editor.insertText(insert);\n // The browser already painted the text; just realign the run offsets.\n this.reindex(blockEl);\n this.lastRevision = this.rev();\n }\n\n /** Re-derive data-off / data-len after a native edit (no node replacement). */\n private reindex(el: HTMLElement) {\n let off = 0;\n for (const child of Array.from(el.children) as HTMLElement[]) {\n if (child.dataset.off == null) continue;\n child.dataset.off = String(off);\n const len = child.dataset.atom != null ? 1 : (child.textContent ?? \"\").length;\n child.dataset.len = String(len);\n off += len;\n }\n }\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport {\n forwardRef,\n useEffect,\n useImperativeHandle,\n useLayoutEffect,\n useRef,\n useState,\n type CSSProperties,\n} from \"react\";\nimport { EditorView } from \"./ce/view\";\nimport { useEditorSnapshot } from \"./hooks\";\nimport type { AtomRenderer, BlockRenderer } from \"./renderers\";\n\nexport interface NoteEditorProps {\n editor: EditorController;\n className?: string;\n style?: CSSProperties;\n /** Max width of the centered content column, in px. */\n maxWidth?: number;\n placeholder?: string;\n autoFocus?: boolean;\n readOnly?: boolean;\n /** Renderers for custom (atomic) block node types. */\n blockRenderers?: Record<string, BlockRenderer>;\n /** Renderers for custom inline atom types. */\n atomRenderers?: Record<string, AtomRenderer>;\n}\n\n/** A viewport-space rectangle (client coordinates). */\nexport interface ViewportRect {\n top: number;\n left: number;\n right: number;\n bottom: number;\n width: number;\n height: number;\n}\n\n/** Imperative handle for building floating UI (slash / selection menus). */\nexport interface NoteEditorHandle {\n focus(): void;\n /** Caret position in viewport coordinates, or null if unavailable. */\n getCaretRect(): { x: number; y: number; height: number } | null;\n /** Bounding box of the current selection in viewport coordinates, or null. */\n getSelectionRect(): ViewportRect | null;\n /** The scrolling element, for scroll-aware positioning. */\n getScrollElement(): HTMLElement | null;\n}\n\nfunction caretClientRect(): DOMRect | null {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0) return null;\n const r = s.getRangeAt(0).cloneRange();\n r.collapse(s.focusNode === r.endContainer && s.focusOffset === r.endOffset ? false : true);\n const rects = r.getClientRects();\n if (rects.length) return rects[rects.length - 1];\n const b = r.getBoundingClientRect();\n return b.height || b.width ? b : null;\n}\n\n/**\n * A contentEditable note editor: the browser owns caret, selection, trackpad,\n * native menus and IME on the live text, while edits are routed through the\n * {@link EditorController} (Y.Doc). A custom caret is drawn on top so it can be\n * branded/animated independently of the (hidden) native one.\n */\nexport const NoteEditor = forwardRef<NoteEditorHandle, NoteEditorProps>(function NoteEditor(\n { editor, className, style, maxWidth = 720, placeholder, autoFocus, readOnly, blockRenderers, atomRenderers },\n ref,\n) {\n const snapshot = useEditorSnapshot(editor);\n const scrollerRef = useRef<HTMLDivElement>(null);\n const contentRef = useRef<HTMLDivElement>(null);\n const viewRef = useRef<EditorView | null>(null);\n const [focused, setFocused] = useState(false);\n const [caret, setCaret] = useState<{ x: number; y: number; h: number } | null>(null);\n\n // Keep the latest renderers reachable without recreating the view.\n const renderersRef = useRef({ blockRenderers, atomRenderers });\n renderersRef.current = { blockRenderers, atomRenderers };\n\n useImperativeHandle(\n ref,\n (): NoteEditorHandle => ({\n focus: () => contentRef.current?.focus(),\n getCaretRect: () => {\n const r = caretClientRect();\n return r ? { x: r.left, y: r.top, height: r.height || 16 } : null;\n },\n getSelectionRect: () => {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || s.isCollapsed) return null;\n const b = s.getRangeAt(0).getBoundingClientRect();\n if (!b.width && !b.height) return null;\n return { top: b.top, left: b.left, right: b.right, bottom: b.bottom, width: b.width, height: b.height };\n },\n getScrollElement: () => scrollerRef.current,\n }),\n [],\n );\n\n // Create the imperative contentEditable view once.\n useEffect(() => {\n const el = contentRef.current;\n if (!el) return;\n const view = new EditorView(el, editor, {\n readOnly,\n renderAtom: (t) => renderersRef.current.atomRenderers?.[t],\n renderBlock: (t) => renderersRef.current.blockRenderers?.[t],\n });\n viewRef.current = view;\n return () => {\n view.destroy();\n viewRef.current = null;\n };\n }, [editor, readOnly]);\n\n // Reconcile the view when the model changes externally (app commands, undo,\n // remote). The view ignores revisions it produced itself (native typing).\n useEffect(() => {\n viewRef.current?.sync();\n }, [snapshot.revision]);\n\n useEffect(() => {\n if (autoFocus) contentRef.current?.focus();\n }, [autoFocus]);\n\n // Position the custom caret from the live DOM selection.\n useEffect(() => {\n const update = () => {\n const content = contentRef.current;\n const s = window.getSelection();\n if (!content || !s || s.rangeCount === 0 || !s.isCollapsed || !content.contains(s.anchorNode)) {\n setCaret(null);\n return;\n }\n const r = caretClientRect();\n const box = content.getBoundingClientRect();\n if (r) setCaret({ x: r.left - box.left, y: r.top - box.top, h: r.height || 18 });\n };\n document.addEventListener(\"selectionchange\", update);\n const ro = new ResizeObserver(update);\n if (contentRef.current) ro.observe(contentRef.current);\n update();\n return () => {\n document.removeEventListener(\"selectionchange\", update);\n ro.disconnect();\n };\n }, []);\n\n // Drive virtualization: keep the controller's width + viewport in sync.\n useLayoutEffect(() => {\n const sc = scrollerRef.current;\n const content = contentRef.current;\n if (!sc || !content) return;\n const sync = () => {\n editor.setWidth(content.clientWidth);\n editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n sync();\n const ro = new ResizeObserver(sync);\n ro.observe(sc);\n ro.observe(content);\n return () => ro.disconnect();\n }, [editor]);\n\n const onScroll = () => {\n const sc = scrollerRef.current;\n if (sc) editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n\n const showCaret = focused && !!caret && !readOnly;\n\n return (\n <div className={`ori-root${className ? ` ${className}` : \"\"}`} style={style}>\n <div className=\"ori-scroller\" ref={scrollerRef} onScroll={onScroll}>\n <div className=\"ori-content\" style={{ maxWidth, marginInline: \"auto\", position: \"relative\" }}>\n <div\n className=\"ori-canvas ori-ce\"\n ref={contentRef}\n onFocus={() => setFocused(true)}\n onBlur={() => setFocused(false)}\n suppressContentEditableWarning\n />\n {showCaret && caret ? (\n <div\n className=\"ori-caret\"\n style={{ position: \"absolute\", left: caret.x, top: caret.y, height: caret.h, pointerEvents: \"none\" }}\n aria-hidden\n />\n ) : null}\n {snapshot.empty && placeholder ? (\n <div className=\"ori-placeholder\" aria-hidden>\n {placeholder}\n </div>\n ) : null}\n </div>\n </div>\n </div>\n );\n});\n","import type { BlockLayout, EditorController, InlineAtom, VisibleBlock } from \"@wingleeio/ori-core\";\nimport { createContext, useContext, type ReactNode } from \"react\";\n\n/** Props for a custom block renderer (atomic nodes: divider, image, …). */\nexport interface BlockRendererProps {\n editor: EditorController;\n block: VisibleBlock;\n /** The block's synthetic layout (atomic blocks: one line, no fragments). */\n layout: BlockLayout;\n}\nexport type BlockRenderer = (props: BlockRendererProps) => ReactNode;\n\n/** Props for a custom inline-atom renderer (mention chip, inline math, …). */\nexport interface AtomRendererProps {\n editor: EditorController;\n atom: InlineAtom;\n}\nexport type AtomRenderer = (props: AtomRendererProps) => ReactNode;\n\nexport interface Renderers {\n blocks: Record<string, BlockRenderer>;\n atoms: Record<string, AtomRenderer>;\n}\n\nconst EMPTY: Renderers = { blocks: {}, atoms: {} };\n\nconst RenderersContext = createContext<Renderers>(EMPTY);\nexport const RenderersProvider = RenderersContext.Provider;\nexport const useRenderers = (): Renderers => useContext(RenderersContext);\n"]}
1
+ {"version":3,"sources":["../src/useEditor.ts","../src/hooks.ts","../src/ce/dom.ts","../src/ce/view.ts","../src/NoteEditor.tsx","../src/renderers.tsx"],"names":["useRef","EditorController","createCanvasMeasurer","useEffect","useSyncExternalStore","textNode","isCollapsed","createRoot","forwardRef","NoteEditor","useState","useImperativeHandle","useLayoutEffect","jsx","jsxs","createContext","useContext"],"mappings":"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","file":"index.cjs","sourcesContent":["import {\n EditorController,\n createCanvasMeasurer,\n type EditorSchema,\n type Measurer,\n type Typography,\n} from \"@wingleeio/ori-core\";\nimport { useEffect, useRef } from \"react\";\nimport type * as Y from \"yjs\";\n\nexport interface UseEditorOptions {\n /** Existing note `Y.Doc`. When switching notes, remount via `key` instead. */\n doc?: Y.Doc;\n typography?: Typography;\n measurer?: Measurer;\n overscan?: number;\n blockSpacing?: number;\n /** Custom block/atom nodes, merged over the built-ins. */\n schema?: Partial<EditorSchema>;\n}\n\n/**\n * Create (once) and own an {@link EditorController} for the lifetime of the\n * component. To switch documents, give the hosting component a `key` so it\n * remounts with a fresh controller.\n *\n * The controller is reconnected on mount and disconnected on unmount rather than\n * destroyed, so React StrictMode's dev mount → unmount → remount cycle reuses the\n * same controller (with all its state) instead of leaving a torn-down one behind.\n */\nexport function useEditor(options: UseEditorOptions = {}): EditorController {\n const ref = useRef<EditorController | null>(null);\n if (ref.current === null) {\n ref.current = new EditorController({\n doc: options.doc,\n measurer: options.measurer ?? createCanvasMeasurer(),\n typography: options.typography,\n overscan: options.overscan,\n blockSpacing: options.blockSpacing,\n schema: options.schema,\n });\n }\n useEffect(() => {\n const editor = ref.current;\n editor?.connect();\n return () => editor?.disconnect();\n }, []);\n return ref.current;\n}\n","import type { EditorController, EditorSnapshot, Marks } from \"@wingleeio/ori-core\";\nimport { useSyncExternalStore } from \"react\";\n\n/** Subscribe a component to the controller's snapshot stream. */\nexport function useEditorSnapshot(editor: EditorController): EditorSnapshot {\n return useSyncExternalStore(editor.subscribe, editor.getSnapshot, editor.getSnapshot);\n}\n\n/**\n * The marks active at the current selection. Recomputed whenever the editor\n * notifies (selection move, edit, or pending-mark toggle), so toolbars stay in\n * sync without their own subscription.\n */\nexport function useActiveMarks(editor: EditorController): Marks {\n const snapshot = useEditorSnapshot(editor);\n // `revision` changes on every notify; reading it ties this to the store.\n void snapshot.revision;\n return editor.getActiveMarks();\n}\n","import type { InlineItem } from \"@wingleeio/ori-core\";\n\n/** CSS.escape with a fallback (jsdom lacks it). */\nexport function esc(s: string): string {\n return typeof CSS !== \"undefined\" && CSS.escape ? CSS.escape(s) : s.replace(/[\"\\\\]/g, \"\\\\$&\");\n}\n\n/**\n * Imperative DOM helpers for the contentEditable view. Each block is a\n * block-level element carrying `data-block-id`; its inline runs are spans\n * carrying `data-off` (their start offset in the block) so DOM positions map\n * back to {blockId, offset} and vice-versa.\n */\n\nexport const BLOCK_SEL = \"[data-block-id]\";\n\nexport function blockElOf(node: Node | null, root: HTMLElement): HTMLElement | null {\n let n: Node | null = node;\n while (n && n !== root) {\n if (n instanceof HTMLElement && n.dataset.blockId) return n;\n n = n.parentNode;\n }\n return null;\n}\n\nfunction spanOf(node: Node | null): HTMLElement | null {\n let n: Node | null = node;\n while (n) {\n if (n instanceof HTMLElement && n.dataset.off != null) return n;\n n = n.parentNode;\n }\n return null;\n}\n\n/** Map a DOM (node, offset) to a {blockId, offset} model position. */\nexport function domToModel(\n root: HTMLElement,\n node: Node | null,\n offset: number,\n): { blockId: string; offset: number } | null {\n const blockEl = blockElOf(node, root);\n if (!blockEl) return null;\n const blockId = blockEl.dataset.blockId as string;\n\n if (node && node.nodeType === Node.TEXT_NODE) {\n const span = spanOf(node);\n const base = span ? Number(span.dataset.off) : 0;\n return { blockId, offset: base + offset };\n }\n\n // node is an element; `offset` is a child index. Resolve via the child spans.\n const el = node as HTMLElement;\n if (el.dataset?.off != null) {\n // selection landed on a span boundary\n return { blockId, offset: Number(el.dataset.off) + (offset > 0 ? spanLen(el) : 0) };\n }\n const kids = Array.from(el.childNodes);\n for (let i = offset; i < kids.length; i++) {\n const k = kids[i];\n if (k instanceof HTMLElement && k.dataset.off != null) return { blockId, offset: Number(k.dataset.off) };\n }\n // past the last span → block end\n let end = 0;\n for (const k of kids) if (k instanceof HTMLElement && k.dataset.off != null) end = Math.max(end, Number(k.dataset.off) + spanLen(k));\n return { blockId, offset: end };\n}\n\nfunction spanLen(span: HTMLElement): number {\n return span.dataset.len != null ? Number(span.dataset.len) : (span.textContent ?? \"\").length;\n}\n\n/** Find the DOM (node, offset) for a {blockId, offset} model position. */\nexport function modelToDom(\n root: HTMLElement,\n blockId: string,\n offset: number,\n): { node: Node; offset: number } | null {\n const blockEl = root.querySelector(`[data-block-id=\"${esc(blockId)}\"]`) as HTMLElement | null;\n if (!blockEl) return null;\n const spans = Array.from(blockEl.querySelectorAll(\"[data-off]\")) as HTMLElement[];\n if (spans.length === 0) {\n return { node: blockEl, offset: 0 }; // empty block\n }\n for (const span of spans) {\n const start = Number(span.dataset.off);\n const len = spanLen(span);\n if (offset <= start + len) {\n if (span.dataset.atom != null) {\n // atom: place before or after it (offset is start or start+1)\n const idx = Array.prototype.indexOf.call(blockEl.childNodes, span);\n return { node: blockEl, offset: offset <= start ? idx : idx + 1 };\n }\n const textNode = span.firstChild ?? span;\n return { node: textNode, offset: Math.max(0, Math.min(offset - start, (textNode.textContent ?? \"\").length)) };\n }\n }\n // past everything → after the last span\n const last = spans[spans.length - 1];\n const textNode = last.firstChild ?? last;\n return { node: textNode, offset: (textNode.textContent ?? \"\").length };\n}\n\nfunction markClass(marks: InlineItem[\"marks\"]): string {\n const m = marks ?? {};\n const cls = [\"ori-frag\"];\n if (m.bold) cls.push(\"ori-m-bold\");\n if (m.italic) cls.push(\"ori-m-italic\");\n if (m.underline) cls.push(\"ori-m-underline\");\n if (m.strike) cls.push(\"ori-m-strike\");\n if (m.code) cls.push(\"ori-frag-code\");\n if (m.link) cls.push(\"ori-frag-link\");\n return cls.join(\" \");\n}\n\n/** Build the inline run DOM for a block (text spans only; atoms handled by the view). */\nexport function buildRun(item: InlineItem): HTMLElement {\n const span = document.createElement(\"span\");\n span.className = markClass(item.marks);\n span.dataset.off = String(item.start);\n span.dataset.len = String(item.text.length);\n span.textContent = item.text;\n return span;\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport { isCollapsed } from \"@wingleeio/ori-core\";\nimport type { ReactNode } from \"react\";\nimport { createRoot, type Root } from \"react-dom/client\";\nimport type { AtomRenderer, BlockRenderer } from \"../renderers\";\nimport { blockElOf, buildRun, domToModel, esc, modelToDom } from \"./dom\";\n\nconst PLACEHOLDER = \"\";\n\nexport interface ViewOptions {\n readOnly?: boolean;\n renderAtom: (type: string) => AtomRenderer | undefined;\n renderBlock: (type: string) => BlockRenderer | undefined;\n}\n\n/**\n * Imperative contentEditable view over an {@link EditorController}. The browser\n * owns caret / selection / trackpad / menus / IME on the live text; we intercept\n * structural + cross-block edits (beforeinput) and route them through the\n * controller, let smooth in-block typing flow natively and read it back, and\n * keep the DOM selection and the controller selection in lock-step.\n */\nexport class EditorView {\n private roots = new Map<HTMLElement, Root>();\n private composing = false;\n private applyingModel = false;\n private detachers: Array<() => void> = [];\n /** The model revision the DOM currently reflects (so external changes — remote\n * edits, app commands — re-render, but our own edits don't clobber the caret). */\n private lastRevision = -1;\n\n constructor(\n private root: HTMLElement,\n private editor: EditorController,\n private opts: ViewOptions,\n ) {\n root.setAttribute(\"contenteditable\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"spellcheck\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"role\", \"textbox\");\n root.setAttribute(\"aria-multiline\", \"true\");\n this.renderBlocks();\n this.lastRevision = this.rev();\n\n const on = <K extends keyof HTMLElementEventMap>(\n t: K,\n h: (e: HTMLElementEventMap[K]) => void,\n o?: AddEventListenerOptions,\n ) => {\n root.addEventListener(t, h as EventListener, o);\n this.detachers.push(() => root.removeEventListener(t, h as EventListener, o));\n };\n on(\"beforeinput\", (e) => this.onBeforeInput(e as InputEvent));\n on(\"input\", () => this.onInput());\n on(\"keydown\", (e) => this.onKeyDown(e as KeyboardEvent));\n on(\"blur\", () => {\n // Clicking outside the editor drops the selection (so a selection toolbar\n // hides). Defer so we can ignore a window/tab blur and focus-preserving\n // clicks (e.g. toolbar buttons that re-focus the editor).\n setTimeout(() => {\n if (document.activeElement === this.root || !document.hasFocus()) return;\n const sel = this.editor.getSelection();\n if (sel && !isCollapsed(sel)) {\n this.editor.collapse(sel.focus);\n this.lastRevision = this.rev();\n }\n }, 0);\n });\n on(\"compositionstart\", () => (this.composing = true));\n on(\"compositionend\", () => {\n this.composing = false;\n this.onInput();\n });\n\n const onSelChange = () => {\n if (this.applyingModel || this.composing) return;\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n // DOM is already the source of truth here — record the revision so the\n // resulting React sync() doesn't write the selection back and collapse it.\n this.lastRevision = this.rev();\n };\n document.addEventListener(\"selectionchange\", onSelChange);\n this.detachers.push(() => document.removeEventListener(\"selectionchange\", onSelChange));\n }\n\n destroy() {\n this.detachers.forEach((d) => d());\n this.roots.forEach((r) => r.unmount());\n this.roots.clear();\n }\n\n focus() {\n this.root.focus();\n }\n\n // --- rendering ---------------------------------------------------------\n\n private rev(): number {\n return this.editor.getSnapshot().revision;\n }\n\n /**\n * Called by React on every model change. Only re-renders when the model moved\n * ahead of what we last drew (an *external* change — app command, undo, remote);\n * our own edits already updated the DOM and must not be clobbered.\n */\n sync() {\n const rev = this.rev();\n if (rev === this.lastRevision) return;\n // Only restore the DOM selection when the *content* changed (app command,\n // undo, remote). A selection-only change is already correct in the DOM, and\n // writing it back would fight (and collapse) the user's native selection.\n const changed = this.renderBlocks();\n if (changed) this.writeSelection();\n this.lastRevision = rev;\n }\n\n /** After a controlled (preventDefault'd) edit: re-render + restore the caret. */\n private commit() {\n this.renderBlocks();\n this.writeSelection();\n this.lastRevision = this.rev();\n }\n\n /** A content signature for a block, so unchanged blocks aren't re-rendered. */\n private sig(id: string): string {\n return this.editor.getBlockType(id) + \"|\" + JSON.stringify(this.editor.getInline(id));\n }\n\n /**\n * Reconcile the DOM to the *visible window* of blocks (virtualization): a top\n * spacer, the windowed blocks, then a bottom spacer — heights from the\n * controller's offscreen measurement. On-screen blocks are reused by id so a\n * caret inside one survives a scroll. Returns true if the DOM was mutated.\n */\n private renderBlocks(): boolean {\n let changed = false;\n const snap = this.editor.getSnapshot();\n const vis = snap.visible;\n const topH = vis.length ? vis[0].top : 0;\n const botH = vis.length\n ? Math.max(0, snap.totalHeight - (vis[vis.length - 1].top + vis[vis.length - 1].height))\n : Math.max(0, snap.totalHeight);\n\n const TOP = \"\u0000top\";\n const BOTTOM = \"\u0000bottom\";\n const want = [TOP, ...vis.map((v) => v.id), BOTTOM];\n const keyOf = (el: Element) => {\n const e = el as HTMLElement;\n return e.dataset.spacer ? \"\u0000\" + e.dataset.spacer : (e.dataset.blockId ?? \"\");\n };\n const have = new Map<string, HTMLElement>();\n for (const c of Array.from(this.root.children)) have.set(keyOf(c), c as HTMLElement);\n\n let prev: HTMLElement | null = null;\n for (const k of want) {\n let el: HTMLElement | undefined = have.get(k);\n if (el) {\n have.delete(k);\n } else {\n el = k === TOP || k === BOTTOM ? this.makeSpacer(k.slice(1)) : this.makeBlock(k);\n changed = true;\n }\n const anchor: ChildNode | null = prev ? prev.nextSibling : this.root.firstChild;\n if (anchor !== el) {\n this.root.insertBefore(el, anchor);\n changed = true;\n }\n prev = el;\n }\n for (const el of have.values()) {\n if (el.dataset.blockId) this.unmountRootsIn(el);\n el.remove();\n changed = true;\n }\n\n const top = this.root.firstElementChild as HTMLElement | null;\n if (top && top.style.height !== `${topH}px`) top.style.height = `${topH}px`;\n const bot = this.root.lastElementChild as HTMLElement | null;\n if (bot && bot.style.height !== `${botH}px`) bot.style.height = `${botH}px`;\n\n for (const vb of vis) {\n const el = this.root.querySelector(`[data-block-id=\"${esc(vb.id)}\"]`) as HTMLElement | null;\n if (!el) continue;\n const sig = this.sig(vb.id);\n if (el.dataset.sig !== sig) {\n el.dataset.sig = sig;\n this.renderBlockInner(el, vb.id);\n changed = true;\n }\n }\n return changed;\n }\n\n private makeBlock(id: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.blockId = id;\n return el;\n }\n\n private makeSpacer(which: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.spacer = which;\n el.setAttribute(\"contenteditable\", \"false\");\n el.setAttribute(\"aria-hidden\", \"true\");\n el.style.userSelect = \"none\";\n el.style.pointerEvents = \"none\";\n return el;\n }\n\n private renderBlockInner(el: HTMLElement, id: string) {\n this.unmountRootsIn(el);\n const type = this.editor.getBlockType(id);\n el.className = `ori-block ori-block-${type}`;\n\n const blockRenderer = this.opts.renderBlock(type);\n if (blockRenderer) {\n el.contentEditable = \"false\";\n el.textContent = \"\";\n const root = createRoot(el);\n root.render(blockRenderer({ editor: this.editor, block: { id, type, index: 0, top: 0, height: 0 }, layout: this.editor.getLayout(id)! }) as ReactNode);\n this.roots.set(el, root);\n return;\n }\n el.contentEditable = \"inherit\";\n el.textContent = \"\";\n const items = this.editor.getInline(id);\n if (items.length === 0) {\n el.appendChild(document.createElement(\"br\")); // keep an empty block selectable\n return;\n }\n for (const item of items) {\n if (item.atom) {\n const span = document.createElement(\"span\");\n span.className = \"ori-atom\";\n span.contentEditable = \"false\";\n span.dataset.atom = \"true\";\n span.dataset.off = String(item.start);\n span.dataset.len = \"1\";\n el.appendChild(span);\n const renderer = this.opts.renderAtom(item.atom.type);\n if (renderer) {\n const r = createRoot(span);\n r.render(renderer({ editor: this.editor, atom: item.atom }) as ReactNode);\n this.roots.set(span, r);\n }\n } else {\n el.appendChild(buildRun(item));\n }\n }\n }\n\n private unmountRootsIn(el: HTMLElement) {\n for (const [node, root] of this.roots) {\n if (el === node || el.contains(node)) {\n root.unmount();\n this.roots.delete(node);\n }\n }\n }\n\n // --- selection ---------------------------------------------------------\n\n private readSelection() {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || !this.root.contains(s.anchorNode)) return null;\n const a = domToModel(this.root, s.anchorNode, s.anchorOffset);\n const f = domToModel(this.root, s.focusNode, s.focusOffset);\n if (!a || !f) return null;\n return { anchor: { blockId: a.blockId, offset: a.offset }, focus: { blockId: f.blockId, offset: f.offset } };\n }\n\n /** Push the controller's selection back into the DOM (after a model op). */\n private writeSelection() {\n const sel = this.editor.getSelection();\n if (!sel) return;\n const a = modelToDom(this.root, sel.anchor.blockId, sel.anchor.offset);\n const f = modelToDom(this.root, sel.focus.blockId, sel.focus.offset);\n if (!a || !f) return;\n const r = document.createRange();\n const s = window.getSelection();\n if (!s) return;\n this.applyingModel = true;\n try {\n r.setStart(a.node, a.offset);\n s.removeAllRanges();\n s.addRange(r);\n s.extend(f.node, f.offset);\n } catch {\n /* node detached mid-reconcile */\n } finally {\n this.applyingModel = false;\n }\n }\n\n /** The block text as the model sees it (atoms collapse to one placeholder). */\n private domBlockText(el: HTMLElement): string {\n let out = \"\";\n for (const child of Array.from(el.childNodes)) {\n if (child instanceof HTMLElement && child.dataset.atom != null) {\n out += PLACEHOLDER;\n } else {\n out += child.textContent ?? \"\";\n }\n }\n return out;\n }\n\n // --- input -------------------------------------------------------------\n\n /** Formatting + history shortcuts (the browser fires these as keydown). */\n private onKeyDown(e: KeyboardEvent) {\n if (this.opts.readOnly) return;\n const mod = e.metaKey || e.ctrlKey;\n if (!mod || e.altKey) return;\n const k = e.key.toLowerCase();\n const mark = ({ b: \"bold\", i: \"italic\", u: \"underline\", e: \"code\" } as const)[k];\n if (mark) {\n e.preventDefault();\n const sel = this.readSelection();\n if (sel) this.editor.setSelection(sel);\n this.editor.toggleMark(mark);\n this.commit();\n } else if (k === \"z\") {\n e.preventDefault();\n if (e.shiftKey) this.editor.redo();\n else this.editor.undo();\n this.commit();\n } else if (k === \"y\") {\n e.preventDefault();\n this.editor.redo();\n this.commit();\n }\n }\n\n private onBeforeInput(e: InputEvent) {\n if (this.opts.readOnly) {\n e.preventDefault();\n return;\n }\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n const collapsed = isCollapsed(sel);\n const startOffset = this.editor.orderedSelection()?.start.offset ?? sel.focus.offset;\n const t = e.inputType;\n\n // Native fast path: collapsed in-block typing / deletion. The browser mutates\n // a single text node; onInput reads it back. Keeps autocorrect/IME native.\n if (collapsed && (t === \"insertText\" || t === \"insertCompositionText\" || t === \"insertReplacementText\")) return;\n if (collapsed && t === \"deleteContentForward\") return;\n if (collapsed && t === \"deleteContentBackward\" && startOffset > 0) return;\n\n // Everything else (structural + cross-block) is handled through the controller.\n const ed = this.editor;\n if (t === \"insertParagraph\") {\n e.preventDefault();\n ed.insertParagraphBreak();\n } else if (t.startsWith(\"delete\")) {\n e.preventDefault();\n if (t === \"deleteContentForward\") ed.deleteForward();\n else ed.deleteBackward();\n } else if (t === \"insertText\" || t === \"insertReplacementText\" || t === \"insertFromPaste\") {\n e.preventDefault();\n const text = e.data ?? e.dataTransfer?.getData(\"text/plain\") ?? \"\";\n if (text) ed.insertText(text);\n } else if (t === \"insertLineBreak\") {\n e.preventDefault();\n ed.insertText(\"\\n\");\n } else {\n return; // let the browser handle anything we don't model\n }\n this.commit();\n }\n\n private onInput() {\n if (this.composing || this.opts.readOnly) return;\n const blockEl = blockElOf(window.getSelection()?.anchorNode ?? null, this.root);\n if (!blockEl) {\n // structure changed under us (browser merged blocks) → full resync\n this.renderBlocks();\n this.lastRevision = this.rev();\n return;\n }\n const id = blockEl.dataset.blockId as string;\n const next = this.domBlockText(blockEl);\n const cur = this.editor.getBlockText(id);\n if (next === cur) return;\n\n // diff → splice through the controller (which infers marks at the caret)\n const max = Math.min(cur.length, next.length);\n let p = 0;\n while (p < max && cur[p] === next[p]) p++;\n let s = 0;\n while (s < max - p && cur[cur.length - 1 - s] === next[next.length - 1 - s]) s++;\n const from = p;\n const to = cur.length - s;\n const insert = next.slice(p, next.length - s);\n this.editor.setSelection({ anchor: { blockId: id, offset: from }, focus: { blockId: id, offset: to } });\n if (to > from) this.editor.deleteBackward();\n if (insert) this.editor.insertText(insert);\n // The browser already painted the text; just realign the run offsets.\n this.reindex(blockEl);\n this.lastRevision = this.rev();\n }\n\n /** Re-derive data-off / data-len after a native edit (no node replacement). */\n private reindex(el: HTMLElement) {\n let off = 0;\n for (const child of Array.from(el.children) as HTMLElement[]) {\n if (child.dataset.off == null) continue;\n child.dataset.off = String(off);\n const len = child.dataset.atom != null ? 1 : (child.textContent ?? \"\").length;\n child.dataset.len = String(len);\n off += len;\n }\n }\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport {\n forwardRef,\n useEffect,\n useImperativeHandle,\n useLayoutEffect,\n useRef,\n useState,\n type CSSProperties,\n type PointerEvent as ReactPointerEvent,\n} from \"react\";\nimport { EditorView } from \"./ce/view\";\nimport { useEditorSnapshot } from \"./hooks\";\nimport type { AtomRenderer, BlockRenderer } from \"./renderers\";\n\nexport interface NoteEditorProps {\n editor: EditorController;\n className?: string;\n style?: CSSProperties;\n /** Max width of the centered content column, in px. */\n maxWidth?: number;\n placeholder?: string;\n autoFocus?: boolean;\n readOnly?: boolean;\n /** Renderers for custom (atomic) block node types. */\n blockRenderers?: Record<string, BlockRenderer>;\n /** Renderers for custom inline atom types. */\n atomRenderers?: Record<string, AtomRenderer>;\n}\n\n/** A viewport-space rectangle (client coordinates). */\nexport interface ViewportRect {\n top: number;\n left: number;\n right: number;\n bottom: number;\n width: number;\n height: number;\n}\n\n/** Imperative handle for building floating UI (slash / selection menus). */\nexport interface NoteEditorHandle {\n focus(): void;\n /** Caret position in viewport coordinates, or null if unavailable. */\n getCaretRect(): { x: number; y: number; height: number } | null;\n /** Bounding box of the current selection in viewport coordinates, or null. */\n getSelectionRect(): ViewportRect | null;\n /** The scrolling element, for scroll-aware positioning. */\n getScrollElement(): HTMLElement | null;\n /**\n * The content overlay element (a positioned layer that scrolls *with* the\n * text). Render floating UI into this with `position: absolute` and\n * content-relative coordinates so it rides the scroll natively instead of\n * trailing it (which causes a fixed-position toolbar to shake on scroll).\n */\n getOverlayElement(): HTMLElement | null;\n}\n\nfunction caretClientRect(): DOMRect | null {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0) return null;\n const r = s.getRangeAt(0).cloneRange();\n r.collapse(s.focusNode === r.endContainer && s.focusOffset === r.endOffset ? false : true);\n const rects = r.getClientRects();\n if (rects.length) return rects[rects.length - 1];\n const b = r.getBoundingClientRect();\n if (b.height || b.width) return b;\n // Empty block (`<br>` only): a collapsed range there has no client rects, so\n // synthesize the caret from the block box + its line metrics. Without this the\n // custom caret would vanish on empty lines (the native caret is hidden).\n const node = r.startContainer;\n const el = (node.nodeType === Node.TEXT_NODE ? node.parentElement : (node as HTMLElement)) ?? null;\n if (!el) return null;\n const eb = el.getBoundingClientRect();\n const cs = getComputedStyle(el);\n const lh = parseFloat(cs.lineHeight) || parseFloat(cs.fontSize) * 1.4 || 18;\n const padL = parseFloat(cs.paddingLeft) || 0;\n const padT = parseFloat(cs.paddingTop) || 0;\n return new DOMRect(eb.left + padL, eb.top + padT, 0, lh);\n}\n\n/**\n * A contentEditable note editor: the browser owns caret, selection, trackpad,\n * native menus and IME on the live text, while edits are routed through the\n * {@link EditorController} (Y.Doc). A custom caret is drawn on top so it can be\n * branded/animated independently of the (hidden) native one.\n */\nexport const NoteEditor = forwardRef<NoteEditorHandle, NoteEditorProps>(function NoteEditor(\n { editor, className, style, maxWidth = 720, placeholder, autoFocus, readOnly, blockRenderers, atomRenderers },\n ref,\n) {\n const snapshot = useEditorSnapshot(editor);\n const scrollerRef = useRef<HTMLDivElement>(null);\n const overlayRef = useRef<HTMLDivElement>(null);\n const contentRef = useRef<HTMLDivElement>(null);\n const viewRef = useRef<EditorView | null>(null);\n const [focused, setFocused] = useState(false);\n const [caret, setCaret] = useState<{ x: number; y: number; h: number } | null>(null);\n\n // Keep the latest renderers reachable without recreating the view.\n const renderersRef = useRef({ blockRenderers, atomRenderers });\n renderersRef.current = { blockRenderers, atomRenderers };\n\n useImperativeHandle(\n ref,\n (): NoteEditorHandle => ({\n focus: () => contentRef.current?.focus(),\n getCaretRect: () => {\n const r = caretClientRect();\n return r ? { x: r.left, y: r.top, height: r.height || 16 } : null;\n },\n getSelectionRect: () => {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || s.isCollapsed) return null;\n const b = s.getRangeAt(0).getBoundingClientRect();\n if (!b.width && !b.height) return null;\n return { top: b.top, left: b.left, right: b.right, bottom: b.bottom, width: b.width, height: b.height };\n },\n getScrollElement: () => scrollerRef.current,\n getOverlayElement: () => overlayRef.current,\n }),\n [],\n );\n\n // Create the imperative contentEditable view once.\n useEffect(() => {\n const el = contentRef.current;\n if (!el) return;\n const view = new EditorView(el, editor, {\n readOnly,\n renderAtom: (t) => renderersRef.current.atomRenderers?.[t],\n renderBlock: (t) => renderersRef.current.blockRenderers?.[t],\n });\n viewRef.current = view;\n return () => {\n view.destroy();\n viewRef.current = null;\n };\n }, [editor, readOnly]);\n\n // Reconcile the view when the model changes externally (app commands, undo,\n // remote). The view ignores revisions it produced itself (native typing).\n useEffect(() => {\n viewRef.current?.sync();\n }, [snapshot.revision]);\n\n useEffect(() => {\n if (autoFocus) contentRef.current?.focus();\n }, [autoFocus]);\n\n // Position the custom caret from the live DOM selection.\n useEffect(() => {\n const update = () => {\n const content = contentRef.current;\n const s = window.getSelection();\n if (!content || !s || s.rangeCount === 0 || !s.isCollapsed || !content.contains(s.anchorNode)) {\n setCaret(null);\n return;\n }\n const r = caretClientRect();\n const box = content.getBoundingClientRect();\n if (r) setCaret({ x: r.left - box.left, y: r.top - box.top, h: r.height || 18 });\n };\n document.addEventListener(\"selectionchange\", update);\n const ro = new ResizeObserver(update);\n if (contentRef.current) ro.observe(contentRef.current);\n update();\n return () => {\n document.removeEventListener(\"selectionchange\", update);\n ro.disconnect();\n };\n }, []);\n\n // Drive virtualization: keep the controller's width + viewport in sync.\n useLayoutEffect(() => {\n const sc = scrollerRef.current;\n const content = contentRef.current;\n if (!sc || !content) return;\n const sync = () => {\n editor.setWidth(content.clientWidth);\n editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n sync();\n const ro = new ResizeObserver(sync);\n ro.observe(sc);\n ro.observe(content);\n return () => ro.disconnect();\n }, [editor]);\n\n const onScroll = () => {\n const sc = scrollerRef.current;\n if (sc) editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n\n // Clicking the empty region below the content should focus the editor and put\n // the caret at the document end (the usual \"click to keep writing\" affordance).\n const onPointerDown = (e: ReactPointerEvent<HTMLDivElement>) => {\n if (readOnly) return;\n // Only act on the editor's own empty surface — not on blocks (the browser\n // places the caret) and not on floating UI (menus) layered into the overlay.\n const t = e.target as HTMLElement;\n const onSurface =\n t === scrollerRef.current ||\n t === overlayRef.current ||\n t.classList.contains(\"ori-ce\") ||\n t.dataset.spacer != null;\n if (!onSurface) return;\n const content = contentRef.current;\n const blocks = content?.querySelectorAll(\"[data-block-id]\");\n const last = blocks && (blocks[blocks.length - 1] as HTMLElement | undefined);\n if (!content || !last) return;\n if (e.clientY <= last.getBoundingClientRect().bottom) return; // beside text, not below\n e.preventDefault();\n content.focus();\n const sel = window.getSelection();\n if (!sel) return;\n const range = document.createRange();\n range.selectNodeContents(last);\n range.collapse(false);\n sel.removeAllRanges();\n sel.addRange(range);\n };\n\n const showCaret = focused && !!caret && !readOnly;\n\n return (\n <div className={`ori-root${className ? ` ${className}` : \"\"}`} style={style}>\n <div className=\"ori-scroller\" ref={scrollerRef} onScroll={onScroll} onPointerDown={onPointerDown}>\n <div className=\"ori-content\" ref={overlayRef} style={{ maxWidth, marginInline: \"auto\", position: \"relative\" }}>\n <div\n className=\"ori-canvas ori-ce\"\n ref={contentRef}\n onFocus={() => setFocused(true)}\n onBlur={() => setFocused(false)}\n suppressContentEditableWarning\n />\n {showCaret && caret ? (\n <div\n className=\"ori-caret\"\n style={{ position: \"absolute\", left: caret.x, top: caret.y, height: caret.h, pointerEvents: \"none\" }}\n aria-hidden\n />\n ) : null}\n {snapshot.empty && placeholder ? (\n <div className=\"ori-placeholder\" aria-hidden>\n {placeholder}\n </div>\n ) : null}\n </div>\n </div>\n </div>\n );\n});\n","import type { BlockLayout, EditorController, InlineAtom, VisibleBlock } from \"@wingleeio/ori-core\";\nimport { createContext, useContext, type ReactNode } from \"react\";\n\n/** Props for a custom block renderer (atomic nodes: divider, image, …). */\nexport interface BlockRendererProps {\n editor: EditorController;\n block: VisibleBlock;\n /** The block's synthetic layout (atomic blocks: one line, no fragments). */\n layout: BlockLayout;\n}\nexport type BlockRenderer = (props: BlockRendererProps) => ReactNode;\n\n/** Props for a custom inline-atom renderer (mention chip, inline math, …). */\nexport interface AtomRendererProps {\n editor: EditorController;\n atom: InlineAtom;\n}\nexport type AtomRenderer = (props: AtomRendererProps) => ReactNode;\n\nexport interface Renderers {\n blocks: Record<string, BlockRenderer>;\n atoms: Record<string, AtomRenderer>;\n}\n\nconst EMPTY: Renderers = { blocks: {}, atoms: {} };\n\nconst RenderersContext = createContext<Renderers>(EMPTY);\nexport const RenderersProvider = RenderersContext.Provider;\nexport const useRenderers = (): Renderers => useContext(RenderersContext);\n"]}
package/dist/index.d.cts CHANGED
@@ -90,6 +90,13 @@ interface NoteEditorHandle {
90
90
  getSelectionRect(): ViewportRect | null;
91
91
  /** The scrolling element, for scroll-aware positioning. */
92
92
  getScrollElement(): HTMLElement | null;
93
+ /**
94
+ * The content overlay element (a positioned layer that scrolls *with* the
95
+ * text). Render floating UI into this with `position: absolute` and
96
+ * content-relative coordinates so it rides the scroll natively instead of
97
+ * trailing it (which causes a fixed-position toolbar to shake on scroll).
98
+ */
99
+ getOverlayElement(): HTMLElement | null;
93
100
  }
94
101
  /**
95
102
  * A contentEditable note editor: the browser owns caret, selection, trackpad,
package/dist/index.d.ts CHANGED
@@ -90,6 +90,13 @@ interface NoteEditorHandle {
90
90
  getSelectionRect(): ViewportRect | null;
91
91
  /** The scrolling element, for scroll-aware positioning. */
92
92
  getScrollElement(): HTMLElement | null;
93
+ /**
94
+ * The content overlay element (a positioned layer that scrolls *with* the
95
+ * text). Render floating UI into this with `position: absolute` and
96
+ * content-relative coordinates so it rides the scroll natively instead of
97
+ * trailing it (which causes a fixed-position toolbar to shake on scroll).
98
+ */
99
+ getOverlayElement(): HTMLElement | null;
93
100
  }
94
101
  /**
95
102
  * A contentEditable note editor: the browser owns caret, selection, trackpad,
package/dist/index.js CHANGED
@@ -150,6 +150,16 @@ var EditorView = class {
150
150
  on("beforeinput", (e) => this.onBeforeInput(e));
151
151
  on("input", () => this.onInput());
152
152
  on("keydown", (e) => this.onKeyDown(e));
153
+ on("blur", () => {
154
+ setTimeout(() => {
155
+ if (document.activeElement === this.root || !document.hasFocus()) return;
156
+ const sel = this.editor.getSelection();
157
+ if (sel && !isCollapsed(sel)) {
158
+ this.editor.collapse(sel.focus);
159
+ this.lastRevision = this.rev();
160
+ }
161
+ }, 0);
162
+ });
153
163
  on("compositionstart", () => this.composing = true);
154
164
  on("compositionend", () => {
155
165
  this.composing = false;
@@ -466,11 +476,21 @@ function caretClientRect() {
466
476
  const rects = r.getClientRects();
467
477
  if (rects.length) return rects[rects.length - 1];
468
478
  const b = r.getBoundingClientRect();
469
- return b.height || b.width ? b : null;
479
+ if (b.height || b.width) return b;
480
+ const node = r.startContainer;
481
+ const el = (node.nodeType === Node.TEXT_NODE ? node.parentElement : node) ?? null;
482
+ if (!el) return null;
483
+ const eb = el.getBoundingClientRect();
484
+ const cs = getComputedStyle(el);
485
+ const lh = parseFloat(cs.lineHeight) || parseFloat(cs.fontSize) * 1.4 || 18;
486
+ const padL = parseFloat(cs.paddingLeft) || 0;
487
+ const padT = parseFloat(cs.paddingTop) || 0;
488
+ return new DOMRect(eb.left + padL, eb.top + padT, 0, lh);
470
489
  }
471
490
  var NoteEditor = forwardRef(function NoteEditor2({ editor, className, style, maxWidth = 720, placeholder, autoFocus, readOnly, blockRenderers, atomRenderers }, ref) {
472
491
  const snapshot = useEditorSnapshot(editor);
473
492
  const scrollerRef = useRef(null);
493
+ const overlayRef = useRef(null);
474
494
  const contentRef = useRef(null);
475
495
  const viewRef = useRef(null);
476
496
  const [focused, setFocused] = useState(false);
@@ -492,7 +512,8 @@ var NoteEditor = forwardRef(function NoteEditor2({ editor, className, style, max
492
512
  if (!b.width && !b.height) return null;
493
513
  return { top: b.top, left: b.left, right: b.right, bottom: b.bottom, width: b.width, height: b.height };
494
514
  },
495
- getScrollElement: () => scrollerRef.current
515
+ getScrollElement: () => scrollerRef.current,
516
+ getOverlayElement: () => overlayRef.current
496
517
  }),
497
518
  []
498
519
  );
@@ -555,8 +576,28 @@ var NoteEditor = forwardRef(function NoteEditor2({ editor, className, style, max
555
576
  const sc = scrollerRef.current;
556
577
  if (sc) editor.setViewport(sc.scrollTop, sc.clientHeight);
557
578
  };
579
+ const onPointerDown = (e) => {
580
+ if (readOnly) return;
581
+ const t = e.target;
582
+ const onSurface = t === scrollerRef.current || t === overlayRef.current || t.classList.contains("ori-ce") || t.dataset.spacer != null;
583
+ if (!onSurface) return;
584
+ const content = contentRef.current;
585
+ const blocks = content?.querySelectorAll("[data-block-id]");
586
+ const last = blocks && blocks[blocks.length - 1];
587
+ if (!content || !last) return;
588
+ if (e.clientY <= last.getBoundingClientRect().bottom) return;
589
+ e.preventDefault();
590
+ content.focus();
591
+ const sel = window.getSelection();
592
+ if (!sel) return;
593
+ const range = document.createRange();
594
+ range.selectNodeContents(last);
595
+ range.collapse(false);
596
+ sel.removeAllRanges();
597
+ sel.addRange(range);
598
+ };
558
599
  const showCaret = focused && !!caret && !readOnly;
559
- return /* @__PURE__ */ jsx("div", { className: `ori-root${className ? ` ${className}` : ""}`, style, children: /* @__PURE__ */ jsx("div", { className: "ori-scroller", ref: scrollerRef, onScroll, children: /* @__PURE__ */ jsxs("div", { className: "ori-content", style: { maxWidth, marginInline: "auto", position: "relative" }, children: [
600
+ return /* @__PURE__ */ jsx("div", { className: `ori-root${className ? ` ${className}` : ""}`, style, children: /* @__PURE__ */ jsx("div", { className: "ori-scroller", ref: scrollerRef, onScroll, onPointerDown, children: /* @__PURE__ */ jsxs("div", { className: "ori-content", ref: overlayRef, style: { maxWidth, marginInline: "auto", position: "relative" }, children: [
560
601
  /* @__PURE__ */ jsx(
561
602
  "div",
562
603
  {
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/useEditor.ts","../src/hooks.ts","../src/ce/dom.ts","../src/ce/view.ts","../src/NoteEditor.tsx","../src/renderers.tsx"],"names":["textNode","NoteEditor","useRef","useEffect"],"mappings":"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","file":"index.js","sourcesContent":["import {\n EditorController,\n createCanvasMeasurer,\n type EditorSchema,\n type Measurer,\n type Typography,\n} from \"@wingleeio/ori-core\";\nimport { useEffect, useRef } from \"react\";\nimport type * as Y from \"yjs\";\n\nexport interface UseEditorOptions {\n /** Existing note `Y.Doc`. When switching notes, remount via `key` instead. */\n doc?: Y.Doc;\n typography?: Typography;\n measurer?: Measurer;\n overscan?: number;\n blockSpacing?: number;\n /** Custom block/atom nodes, merged over the built-ins. */\n schema?: Partial<EditorSchema>;\n}\n\n/**\n * Create (once) and own an {@link EditorController} for the lifetime of the\n * component. To switch documents, give the hosting component a `key` so it\n * remounts with a fresh controller.\n *\n * The controller is reconnected on mount and disconnected on unmount rather than\n * destroyed, so React StrictMode's dev mount → unmount → remount cycle reuses the\n * same controller (with all its state) instead of leaving a torn-down one behind.\n */\nexport function useEditor(options: UseEditorOptions = {}): EditorController {\n const ref = useRef<EditorController | null>(null);\n if (ref.current === null) {\n ref.current = new EditorController({\n doc: options.doc,\n measurer: options.measurer ?? createCanvasMeasurer(),\n typography: options.typography,\n overscan: options.overscan,\n blockSpacing: options.blockSpacing,\n schema: options.schema,\n });\n }\n useEffect(() => {\n const editor = ref.current;\n editor?.connect();\n return () => editor?.disconnect();\n }, []);\n return ref.current;\n}\n","import type { EditorController, EditorSnapshot, Marks } from \"@wingleeio/ori-core\";\nimport { useSyncExternalStore } from \"react\";\n\n/** Subscribe a component to the controller's snapshot stream. */\nexport function useEditorSnapshot(editor: EditorController): EditorSnapshot {\n return useSyncExternalStore(editor.subscribe, editor.getSnapshot, editor.getSnapshot);\n}\n\n/**\n * The marks active at the current selection. Recomputed whenever the editor\n * notifies (selection move, edit, or pending-mark toggle), so toolbars stay in\n * sync without their own subscription.\n */\nexport function useActiveMarks(editor: EditorController): Marks {\n const snapshot = useEditorSnapshot(editor);\n // `revision` changes on every notify; reading it ties this to the store.\n void snapshot.revision;\n return editor.getActiveMarks();\n}\n","import type { InlineItem } from \"@wingleeio/ori-core\";\n\n/** CSS.escape with a fallback (jsdom lacks it). */\nexport function esc(s: string): string {\n return typeof CSS !== \"undefined\" && CSS.escape ? CSS.escape(s) : s.replace(/[\"\\\\]/g, \"\\\\$&\");\n}\n\n/**\n * Imperative DOM helpers for the contentEditable view. Each block is a\n * block-level element carrying `data-block-id`; its inline runs are spans\n * carrying `data-off` (their start offset in the block) so DOM positions map\n * back to {blockId, offset} and vice-versa.\n */\n\nexport const BLOCK_SEL = \"[data-block-id]\";\n\nexport function blockElOf(node: Node | null, root: HTMLElement): HTMLElement | null {\n let n: Node | null = node;\n while (n && n !== root) {\n if (n instanceof HTMLElement && n.dataset.blockId) return n;\n n = n.parentNode;\n }\n return null;\n}\n\nfunction spanOf(node: Node | null): HTMLElement | null {\n let n: Node | null = node;\n while (n) {\n if (n instanceof HTMLElement && n.dataset.off != null) return n;\n n = n.parentNode;\n }\n return null;\n}\n\n/** Map a DOM (node, offset) to a {blockId, offset} model position. */\nexport function domToModel(\n root: HTMLElement,\n node: Node | null,\n offset: number,\n): { blockId: string; offset: number } | null {\n const blockEl = blockElOf(node, root);\n if (!blockEl) return null;\n const blockId = blockEl.dataset.blockId as string;\n\n if (node && node.nodeType === Node.TEXT_NODE) {\n const span = spanOf(node);\n const base = span ? Number(span.dataset.off) : 0;\n return { blockId, offset: base + offset };\n }\n\n // node is an element; `offset` is a child index. Resolve via the child spans.\n const el = node as HTMLElement;\n if (el.dataset?.off != null) {\n // selection landed on a span boundary\n return { blockId, offset: Number(el.dataset.off) + (offset > 0 ? spanLen(el) : 0) };\n }\n const kids = Array.from(el.childNodes);\n for (let i = offset; i < kids.length; i++) {\n const k = kids[i];\n if (k instanceof HTMLElement && k.dataset.off != null) return { blockId, offset: Number(k.dataset.off) };\n }\n // past the last span → block end\n let end = 0;\n for (const k of kids) if (k instanceof HTMLElement && k.dataset.off != null) end = Math.max(end, Number(k.dataset.off) + spanLen(k));\n return { blockId, offset: end };\n}\n\nfunction spanLen(span: HTMLElement): number {\n return span.dataset.len != null ? Number(span.dataset.len) : (span.textContent ?? \"\").length;\n}\n\n/** Find the DOM (node, offset) for a {blockId, offset} model position. */\nexport function modelToDom(\n root: HTMLElement,\n blockId: string,\n offset: number,\n): { node: Node; offset: number } | null {\n const blockEl = root.querySelector(`[data-block-id=\"${esc(blockId)}\"]`) as HTMLElement | null;\n if (!blockEl) return null;\n const spans = Array.from(blockEl.querySelectorAll(\"[data-off]\")) as HTMLElement[];\n if (spans.length === 0) {\n return { node: blockEl, offset: 0 }; // empty block\n }\n for (const span of spans) {\n const start = Number(span.dataset.off);\n const len = spanLen(span);\n if (offset <= start + len) {\n if (span.dataset.atom != null) {\n // atom: place before or after it (offset is start or start+1)\n const idx = Array.prototype.indexOf.call(blockEl.childNodes, span);\n return { node: blockEl, offset: offset <= start ? idx : idx + 1 };\n }\n const textNode = span.firstChild ?? span;\n return { node: textNode, offset: Math.max(0, Math.min(offset - start, (textNode.textContent ?? \"\").length)) };\n }\n }\n // past everything → after the last span\n const last = spans[spans.length - 1];\n const textNode = last.firstChild ?? last;\n return { node: textNode, offset: (textNode.textContent ?? \"\").length };\n}\n\nfunction markClass(marks: InlineItem[\"marks\"]): string {\n const m = marks ?? {};\n const cls = [\"ori-frag\"];\n if (m.bold) cls.push(\"ori-m-bold\");\n if (m.italic) cls.push(\"ori-m-italic\");\n if (m.underline) cls.push(\"ori-m-underline\");\n if (m.strike) cls.push(\"ori-m-strike\");\n if (m.code) cls.push(\"ori-frag-code\");\n if (m.link) cls.push(\"ori-frag-link\");\n return cls.join(\" \");\n}\n\n/** Build the inline run DOM for a block (text spans only; atoms handled by the view). */\nexport function buildRun(item: InlineItem): HTMLElement {\n const span = document.createElement(\"span\");\n span.className = markClass(item.marks);\n span.dataset.off = String(item.start);\n span.dataset.len = String(item.text.length);\n span.textContent = item.text;\n return span;\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport { isCollapsed } from \"@wingleeio/ori-core\";\nimport type { ReactNode } from \"react\";\nimport { createRoot, type Root } from \"react-dom/client\";\nimport type { AtomRenderer, BlockRenderer } from \"../renderers\";\nimport { blockElOf, buildRun, domToModel, esc, modelToDom } from \"./dom\";\n\nconst PLACEHOLDER = \"\";\n\nexport interface ViewOptions {\n readOnly?: boolean;\n renderAtom: (type: string) => AtomRenderer | undefined;\n renderBlock: (type: string) => BlockRenderer | undefined;\n}\n\n/**\n * Imperative contentEditable view over an {@link EditorController}. The browser\n * owns caret / selection / trackpad / menus / IME on the live text; we intercept\n * structural + cross-block edits (beforeinput) and route them through the\n * controller, let smooth in-block typing flow natively and read it back, and\n * keep the DOM selection and the controller selection in lock-step.\n */\nexport class EditorView {\n private roots = new Map<HTMLElement, Root>();\n private composing = false;\n private applyingModel = false;\n private detachers: Array<() => void> = [];\n /** The model revision the DOM currently reflects (so external changes — remote\n * edits, app commands — re-render, but our own edits don't clobber the caret). */\n private lastRevision = -1;\n\n constructor(\n private root: HTMLElement,\n private editor: EditorController,\n private opts: ViewOptions,\n ) {\n root.setAttribute(\"contenteditable\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"spellcheck\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"role\", \"textbox\");\n root.setAttribute(\"aria-multiline\", \"true\");\n this.renderBlocks();\n this.lastRevision = this.rev();\n\n const on = <K extends keyof HTMLElementEventMap>(\n t: K,\n h: (e: HTMLElementEventMap[K]) => void,\n o?: AddEventListenerOptions,\n ) => {\n root.addEventListener(t, h as EventListener, o);\n this.detachers.push(() => root.removeEventListener(t, h as EventListener, o));\n };\n on(\"beforeinput\", (e) => this.onBeforeInput(e as InputEvent));\n on(\"input\", () => this.onInput());\n on(\"keydown\", (e) => this.onKeyDown(e as KeyboardEvent));\n on(\"compositionstart\", () => (this.composing = true));\n on(\"compositionend\", () => {\n this.composing = false;\n this.onInput();\n });\n\n const onSelChange = () => {\n if (this.applyingModel || this.composing) return;\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n // DOM is already the source of truth here — record the revision so the\n // resulting React sync() doesn't write the selection back and collapse it.\n this.lastRevision = this.rev();\n };\n document.addEventListener(\"selectionchange\", onSelChange);\n this.detachers.push(() => document.removeEventListener(\"selectionchange\", onSelChange));\n }\n\n destroy() {\n this.detachers.forEach((d) => d());\n this.roots.forEach((r) => r.unmount());\n this.roots.clear();\n }\n\n focus() {\n this.root.focus();\n }\n\n // --- rendering ---------------------------------------------------------\n\n private rev(): number {\n return this.editor.getSnapshot().revision;\n }\n\n /**\n * Called by React on every model change. Only re-renders when the model moved\n * ahead of what we last drew (an *external* change — app command, undo, remote);\n * our own edits already updated the DOM and must not be clobbered.\n */\n sync() {\n const rev = this.rev();\n if (rev === this.lastRevision) return;\n // Only restore the DOM selection when the *content* changed (app command,\n // undo, remote). A selection-only change is already correct in the DOM, and\n // writing it back would fight (and collapse) the user's native selection.\n const changed = this.renderBlocks();\n if (changed) this.writeSelection();\n this.lastRevision = rev;\n }\n\n /** After a controlled (preventDefault'd) edit: re-render + restore the caret. */\n private commit() {\n this.renderBlocks();\n this.writeSelection();\n this.lastRevision = this.rev();\n }\n\n /** A content signature for a block, so unchanged blocks aren't re-rendered. */\n private sig(id: string): string {\n return this.editor.getBlockType(id) + \"|\" + JSON.stringify(this.editor.getInline(id));\n }\n\n /**\n * Reconcile the DOM to the *visible window* of blocks (virtualization): a top\n * spacer, the windowed blocks, then a bottom spacer — heights from the\n * controller's offscreen measurement. On-screen blocks are reused by id so a\n * caret inside one survives a scroll. Returns true if the DOM was mutated.\n */\n private renderBlocks(): boolean {\n let changed = false;\n const snap = this.editor.getSnapshot();\n const vis = snap.visible;\n const topH = vis.length ? vis[0].top : 0;\n const botH = vis.length\n ? Math.max(0, snap.totalHeight - (vis[vis.length - 1].top + vis[vis.length - 1].height))\n : Math.max(0, snap.totalHeight);\n\n const TOP = \"\u0000top\";\n const BOTTOM = \"\u0000bottom\";\n const want = [TOP, ...vis.map((v) => v.id), BOTTOM];\n const keyOf = (el: Element) => {\n const e = el as HTMLElement;\n return e.dataset.spacer ? \"\u0000\" + e.dataset.spacer : (e.dataset.blockId ?? \"\");\n };\n const have = new Map<string, HTMLElement>();\n for (const c of Array.from(this.root.children)) have.set(keyOf(c), c as HTMLElement);\n\n let prev: HTMLElement | null = null;\n for (const k of want) {\n let el: HTMLElement | undefined = have.get(k);\n if (el) {\n have.delete(k);\n } else {\n el = k === TOP || k === BOTTOM ? this.makeSpacer(k.slice(1)) : this.makeBlock(k);\n changed = true;\n }\n const anchor: ChildNode | null = prev ? prev.nextSibling : this.root.firstChild;\n if (anchor !== el) {\n this.root.insertBefore(el, anchor);\n changed = true;\n }\n prev = el;\n }\n for (const el of have.values()) {\n if (el.dataset.blockId) this.unmountRootsIn(el);\n el.remove();\n changed = true;\n }\n\n const top = this.root.firstElementChild as HTMLElement | null;\n if (top && top.style.height !== `${topH}px`) top.style.height = `${topH}px`;\n const bot = this.root.lastElementChild as HTMLElement | null;\n if (bot && bot.style.height !== `${botH}px`) bot.style.height = `${botH}px`;\n\n for (const vb of vis) {\n const el = this.root.querySelector(`[data-block-id=\"${esc(vb.id)}\"]`) as HTMLElement | null;\n if (!el) continue;\n const sig = this.sig(vb.id);\n if (el.dataset.sig !== sig) {\n el.dataset.sig = sig;\n this.renderBlockInner(el, vb.id);\n changed = true;\n }\n }\n return changed;\n }\n\n private makeBlock(id: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.blockId = id;\n return el;\n }\n\n private makeSpacer(which: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.spacer = which;\n el.setAttribute(\"contenteditable\", \"false\");\n el.setAttribute(\"aria-hidden\", \"true\");\n el.style.userSelect = \"none\";\n el.style.pointerEvents = \"none\";\n return el;\n }\n\n private renderBlockInner(el: HTMLElement, id: string) {\n this.unmountRootsIn(el);\n const type = this.editor.getBlockType(id);\n el.className = `ori-block ori-block-${type}`;\n\n const blockRenderer = this.opts.renderBlock(type);\n if (blockRenderer) {\n el.contentEditable = \"false\";\n el.textContent = \"\";\n const root = createRoot(el);\n root.render(blockRenderer({ editor: this.editor, block: { id, type, index: 0, top: 0, height: 0 }, layout: this.editor.getLayout(id)! }) as ReactNode);\n this.roots.set(el, root);\n return;\n }\n el.contentEditable = \"inherit\";\n el.textContent = \"\";\n const items = this.editor.getInline(id);\n if (items.length === 0) {\n el.appendChild(document.createElement(\"br\")); // keep an empty block selectable\n return;\n }\n for (const item of items) {\n if (item.atom) {\n const span = document.createElement(\"span\");\n span.className = \"ori-atom\";\n span.contentEditable = \"false\";\n span.dataset.atom = \"true\";\n span.dataset.off = String(item.start);\n span.dataset.len = \"1\";\n el.appendChild(span);\n const renderer = this.opts.renderAtom(item.atom.type);\n if (renderer) {\n const r = createRoot(span);\n r.render(renderer({ editor: this.editor, atom: item.atom }) as ReactNode);\n this.roots.set(span, r);\n }\n } else {\n el.appendChild(buildRun(item));\n }\n }\n }\n\n private unmountRootsIn(el: HTMLElement) {\n for (const [node, root] of this.roots) {\n if (el === node || el.contains(node)) {\n root.unmount();\n this.roots.delete(node);\n }\n }\n }\n\n // --- selection ---------------------------------------------------------\n\n private readSelection() {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || !this.root.contains(s.anchorNode)) return null;\n const a = domToModel(this.root, s.anchorNode, s.anchorOffset);\n const f = domToModel(this.root, s.focusNode, s.focusOffset);\n if (!a || !f) return null;\n return { anchor: { blockId: a.blockId, offset: a.offset }, focus: { blockId: f.blockId, offset: f.offset } };\n }\n\n /** Push the controller's selection back into the DOM (after a model op). */\n private writeSelection() {\n const sel = this.editor.getSelection();\n if (!sel) return;\n const a = modelToDom(this.root, sel.anchor.blockId, sel.anchor.offset);\n const f = modelToDom(this.root, sel.focus.blockId, sel.focus.offset);\n if (!a || !f) return;\n const r = document.createRange();\n const s = window.getSelection();\n if (!s) return;\n this.applyingModel = true;\n try {\n r.setStart(a.node, a.offset);\n s.removeAllRanges();\n s.addRange(r);\n s.extend(f.node, f.offset);\n } catch {\n /* node detached mid-reconcile */\n } finally {\n this.applyingModel = false;\n }\n }\n\n /** The block text as the model sees it (atoms collapse to one placeholder). */\n private domBlockText(el: HTMLElement): string {\n let out = \"\";\n for (const child of Array.from(el.childNodes)) {\n if (child instanceof HTMLElement && child.dataset.atom != null) {\n out += PLACEHOLDER;\n } else {\n out += child.textContent ?? \"\";\n }\n }\n return out;\n }\n\n // --- input -------------------------------------------------------------\n\n /** Formatting + history shortcuts (the browser fires these as keydown). */\n private onKeyDown(e: KeyboardEvent) {\n if (this.opts.readOnly) return;\n const mod = e.metaKey || e.ctrlKey;\n if (!mod || e.altKey) return;\n const k = e.key.toLowerCase();\n const mark = ({ b: \"bold\", i: \"italic\", u: \"underline\", e: \"code\" } as const)[k];\n if (mark) {\n e.preventDefault();\n const sel = this.readSelection();\n if (sel) this.editor.setSelection(sel);\n this.editor.toggleMark(mark);\n this.commit();\n } else if (k === \"z\") {\n e.preventDefault();\n if (e.shiftKey) this.editor.redo();\n else this.editor.undo();\n this.commit();\n } else if (k === \"y\") {\n e.preventDefault();\n this.editor.redo();\n this.commit();\n }\n }\n\n private onBeforeInput(e: InputEvent) {\n if (this.opts.readOnly) {\n e.preventDefault();\n return;\n }\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n const collapsed = isCollapsed(sel);\n const startOffset = this.editor.orderedSelection()?.start.offset ?? sel.focus.offset;\n const t = e.inputType;\n\n // Native fast path: collapsed in-block typing / deletion. The browser mutates\n // a single text node; onInput reads it back. Keeps autocorrect/IME native.\n if (collapsed && (t === \"insertText\" || t === \"insertCompositionText\" || t === \"insertReplacementText\")) return;\n if (collapsed && t === \"deleteContentForward\") return;\n if (collapsed && t === \"deleteContentBackward\" && startOffset > 0) return;\n\n // Everything else (structural + cross-block) is handled through the controller.\n const ed = this.editor;\n if (t === \"insertParagraph\") {\n e.preventDefault();\n ed.insertParagraphBreak();\n } else if (t.startsWith(\"delete\")) {\n e.preventDefault();\n if (t === \"deleteContentForward\") ed.deleteForward();\n else ed.deleteBackward();\n } else if (t === \"insertText\" || t === \"insertReplacementText\" || t === \"insertFromPaste\") {\n e.preventDefault();\n const text = e.data ?? e.dataTransfer?.getData(\"text/plain\") ?? \"\";\n if (text) ed.insertText(text);\n } else if (t === \"insertLineBreak\") {\n e.preventDefault();\n ed.insertText(\"\\n\");\n } else {\n return; // let the browser handle anything we don't model\n }\n this.commit();\n }\n\n private onInput() {\n if (this.composing || this.opts.readOnly) return;\n const blockEl = blockElOf(window.getSelection()?.anchorNode ?? null, this.root);\n if (!blockEl) {\n // structure changed under us (browser merged blocks) → full resync\n this.renderBlocks();\n this.lastRevision = this.rev();\n return;\n }\n const id = blockEl.dataset.blockId as string;\n const next = this.domBlockText(blockEl);\n const cur = this.editor.getBlockText(id);\n if (next === cur) return;\n\n // diff → splice through the controller (which infers marks at the caret)\n const max = Math.min(cur.length, next.length);\n let p = 0;\n while (p < max && cur[p] === next[p]) p++;\n let s = 0;\n while (s < max - p && cur[cur.length - 1 - s] === next[next.length - 1 - s]) s++;\n const from = p;\n const to = cur.length - s;\n const insert = next.slice(p, next.length - s);\n this.editor.setSelection({ anchor: { blockId: id, offset: from }, focus: { blockId: id, offset: to } });\n if (to > from) this.editor.deleteBackward();\n if (insert) this.editor.insertText(insert);\n // The browser already painted the text; just realign the run offsets.\n this.reindex(blockEl);\n this.lastRevision = this.rev();\n }\n\n /** Re-derive data-off / data-len after a native edit (no node replacement). */\n private reindex(el: HTMLElement) {\n let off = 0;\n for (const child of Array.from(el.children) as HTMLElement[]) {\n if (child.dataset.off == null) continue;\n child.dataset.off = String(off);\n const len = child.dataset.atom != null ? 1 : (child.textContent ?? \"\").length;\n child.dataset.len = String(len);\n off += len;\n }\n }\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport {\n forwardRef,\n useEffect,\n useImperativeHandle,\n useLayoutEffect,\n useRef,\n useState,\n type CSSProperties,\n} from \"react\";\nimport { EditorView } from \"./ce/view\";\nimport { useEditorSnapshot } from \"./hooks\";\nimport type { AtomRenderer, BlockRenderer } from \"./renderers\";\n\nexport interface NoteEditorProps {\n editor: EditorController;\n className?: string;\n style?: CSSProperties;\n /** Max width of the centered content column, in px. */\n maxWidth?: number;\n placeholder?: string;\n autoFocus?: boolean;\n readOnly?: boolean;\n /** Renderers for custom (atomic) block node types. */\n blockRenderers?: Record<string, BlockRenderer>;\n /** Renderers for custom inline atom types. */\n atomRenderers?: Record<string, AtomRenderer>;\n}\n\n/** A viewport-space rectangle (client coordinates). */\nexport interface ViewportRect {\n top: number;\n left: number;\n right: number;\n bottom: number;\n width: number;\n height: number;\n}\n\n/** Imperative handle for building floating UI (slash / selection menus). */\nexport interface NoteEditorHandle {\n focus(): void;\n /** Caret position in viewport coordinates, or null if unavailable. */\n getCaretRect(): { x: number; y: number; height: number } | null;\n /** Bounding box of the current selection in viewport coordinates, or null. */\n getSelectionRect(): ViewportRect | null;\n /** The scrolling element, for scroll-aware positioning. */\n getScrollElement(): HTMLElement | null;\n}\n\nfunction caretClientRect(): DOMRect | null {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0) return null;\n const r = s.getRangeAt(0).cloneRange();\n r.collapse(s.focusNode === r.endContainer && s.focusOffset === r.endOffset ? false : true);\n const rects = r.getClientRects();\n if (rects.length) return rects[rects.length - 1];\n const b = r.getBoundingClientRect();\n return b.height || b.width ? b : null;\n}\n\n/**\n * A contentEditable note editor: the browser owns caret, selection, trackpad,\n * native menus and IME on the live text, while edits are routed through the\n * {@link EditorController} (Y.Doc). A custom caret is drawn on top so it can be\n * branded/animated independently of the (hidden) native one.\n */\nexport const NoteEditor = forwardRef<NoteEditorHandle, NoteEditorProps>(function NoteEditor(\n { editor, className, style, maxWidth = 720, placeholder, autoFocus, readOnly, blockRenderers, atomRenderers },\n ref,\n) {\n const snapshot = useEditorSnapshot(editor);\n const scrollerRef = useRef<HTMLDivElement>(null);\n const contentRef = useRef<HTMLDivElement>(null);\n const viewRef = useRef<EditorView | null>(null);\n const [focused, setFocused] = useState(false);\n const [caret, setCaret] = useState<{ x: number; y: number; h: number } | null>(null);\n\n // Keep the latest renderers reachable without recreating the view.\n const renderersRef = useRef({ blockRenderers, atomRenderers });\n renderersRef.current = { blockRenderers, atomRenderers };\n\n useImperativeHandle(\n ref,\n (): NoteEditorHandle => ({\n focus: () => contentRef.current?.focus(),\n getCaretRect: () => {\n const r = caretClientRect();\n return r ? { x: r.left, y: r.top, height: r.height || 16 } : null;\n },\n getSelectionRect: () => {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || s.isCollapsed) return null;\n const b = s.getRangeAt(0).getBoundingClientRect();\n if (!b.width && !b.height) return null;\n return { top: b.top, left: b.left, right: b.right, bottom: b.bottom, width: b.width, height: b.height };\n },\n getScrollElement: () => scrollerRef.current,\n }),\n [],\n );\n\n // Create the imperative contentEditable view once.\n useEffect(() => {\n const el = contentRef.current;\n if (!el) return;\n const view = new EditorView(el, editor, {\n readOnly,\n renderAtom: (t) => renderersRef.current.atomRenderers?.[t],\n renderBlock: (t) => renderersRef.current.blockRenderers?.[t],\n });\n viewRef.current = view;\n return () => {\n view.destroy();\n viewRef.current = null;\n };\n }, [editor, readOnly]);\n\n // Reconcile the view when the model changes externally (app commands, undo,\n // remote). The view ignores revisions it produced itself (native typing).\n useEffect(() => {\n viewRef.current?.sync();\n }, [snapshot.revision]);\n\n useEffect(() => {\n if (autoFocus) contentRef.current?.focus();\n }, [autoFocus]);\n\n // Position the custom caret from the live DOM selection.\n useEffect(() => {\n const update = () => {\n const content = contentRef.current;\n const s = window.getSelection();\n if (!content || !s || s.rangeCount === 0 || !s.isCollapsed || !content.contains(s.anchorNode)) {\n setCaret(null);\n return;\n }\n const r = caretClientRect();\n const box = content.getBoundingClientRect();\n if (r) setCaret({ x: r.left - box.left, y: r.top - box.top, h: r.height || 18 });\n };\n document.addEventListener(\"selectionchange\", update);\n const ro = new ResizeObserver(update);\n if (contentRef.current) ro.observe(contentRef.current);\n update();\n return () => {\n document.removeEventListener(\"selectionchange\", update);\n ro.disconnect();\n };\n }, []);\n\n // Drive virtualization: keep the controller's width + viewport in sync.\n useLayoutEffect(() => {\n const sc = scrollerRef.current;\n const content = contentRef.current;\n if (!sc || !content) return;\n const sync = () => {\n editor.setWidth(content.clientWidth);\n editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n sync();\n const ro = new ResizeObserver(sync);\n ro.observe(sc);\n ro.observe(content);\n return () => ro.disconnect();\n }, [editor]);\n\n const onScroll = () => {\n const sc = scrollerRef.current;\n if (sc) editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n\n const showCaret = focused && !!caret && !readOnly;\n\n return (\n <div className={`ori-root${className ? ` ${className}` : \"\"}`} style={style}>\n <div className=\"ori-scroller\" ref={scrollerRef} onScroll={onScroll}>\n <div className=\"ori-content\" style={{ maxWidth, marginInline: \"auto\", position: \"relative\" }}>\n <div\n className=\"ori-canvas ori-ce\"\n ref={contentRef}\n onFocus={() => setFocused(true)}\n onBlur={() => setFocused(false)}\n suppressContentEditableWarning\n />\n {showCaret && caret ? (\n <div\n className=\"ori-caret\"\n style={{ position: \"absolute\", left: caret.x, top: caret.y, height: caret.h, pointerEvents: \"none\" }}\n aria-hidden\n />\n ) : null}\n {snapshot.empty && placeholder ? (\n <div className=\"ori-placeholder\" aria-hidden>\n {placeholder}\n </div>\n ) : null}\n </div>\n </div>\n </div>\n );\n});\n","import type { BlockLayout, EditorController, InlineAtom, VisibleBlock } from \"@wingleeio/ori-core\";\nimport { createContext, useContext, type ReactNode } from \"react\";\n\n/** Props for a custom block renderer (atomic nodes: divider, image, …). */\nexport interface BlockRendererProps {\n editor: EditorController;\n block: VisibleBlock;\n /** The block's synthetic layout (atomic blocks: one line, no fragments). */\n layout: BlockLayout;\n}\nexport type BlockRenderer = (props: BlockRendererProps) => ReactNode;\n\n/** Props for a custom inline-atom renderer (mention chip, inline math, …). */\nexport interface AtomRendererProps {\n editor: EditorController;\n atom: InlineAtom;\n}\nexport type AtomRenderer = (props: AtomRendererProps) => ReactNode;\n\nexport interface Renderers {\n blocks: Record<string, BlockRenderer>;\n atoms: Record<string, AtomRenderer>;\n}\n\nconst EMPTY: Renderers = { blocks: {}, atoms: {} };\n\nconst RenderersContext = createContext<Renderers>(EMPTY);\nexport const RenderersProvider = RenderersContext.Provider;\nexport const useRenderers = (): Renderers => useContext(RenderersContext);\n"]}
1
+ {"version":3,"sources":["../src/useEditor.ts","../src/hooks.ts","../src/ce/dom.ts","../src/ce/view.ts","../src/NoteEditor.tsx","../src/renderers.tsx"],"names":["textNode","NoteEditor","useRef","useEffect"],"mappings":"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","file":"index.js","sourcesContent":["import {\n EditorController,\n createCanvasMeasurer,\n type EditorSchema,\n type Measurer,\n type Typography,\n} from \"@wingleeio/ori-core\";\nimport { useEffect, useRef } from \"react\";\nimport type * as Y from \"yjs\";\n\nexport interface UseEditorOptions {\n /** Existing note `Y.Doc`. When switching notes, remount via `key` instead. */\n doc?: Y.Doc;\n typography?: Typography;\n measurer?: Measurer;\n overscan?: number;\n blockSpacing?: number;\n /** Custom block/atom nodes, merged over the built-ins. */\n schema?: Partial<EditorSchema>;\n}\n\n/**\n * Create (once) and own an {@link EditorController} for the lifetime of the\n * component. To switch documents, give the hosting component a `key` so it\n * remounts with a fresh controller.\n *\n * The controller is reconnected on mount and disconnected on unmount rather than\n * destroyed, so React StrictMode's dev mount → unmount → remount cycle reuses the\n * same controller (with all its state) instead of leaving a torn-down one behind.\n */\nexport function useEditor(options: UseEditorOptions = {}): EditorController {\n const ref = useRef<EditorController | null>(null);\n if (ref.current === null) {\n ref.current = new EditorController({\n doc: options.doc,\n measurer: options.measurer ?? createCanvasMeasurer(),\n typography: options.typography,\n overscan: options.overscan,\n blockSpacing: options.blockSpacing,\n schema: options.schema,\n });\n }\n useEffect(() => {\n const editor = ref.current;\n editor?.connect();\n return () => editor?.disconnect();\n }, []);\n return ref.current;\n}\n","import type { EditorController, EditorSnapshot, Marks } from \"@wingleeio/ori-core\";\nimport { useSyncExternalStore } from \"react\";\n\n/** Subscribe a component to the controller's snapshot stream. */\nexport function useEditorSnapshot(editor: EditorController): EditorSnapshot {\n return useSyncExternalStore(editor.subscribe, editor.getSnapshot, editor.getSnapshot);\n}\n\n/**\n * The marks active at the current selection. Recomputed whenever the editor\n * notifies (selection move, edit, or pending-mark toggle), so toolbars stay in\n * sync without their own subscription.\n */\nexport function useActiveMarks(editor: EditorController): Marks {\n const snapshot = useEditorSnapshot(editor);\n // `revision` changes on every notify; reading it ties this to the store.\n void snapshot.revision;\n return editor.getActiveMarks();\n}\n","import type { InlineItem } from \"@wingleeio/ori-core\";\n\n/** CSS.escape with a fallback (jsdom lacks it). */\nexport function esc(s: string): string {\n return typeof CSS !== \"undefined\" && CSS.escape ? CSS.escape(s) : s.replace(/[\"\\\\]/g, \"\\\\$&\");\n}\n\n/**\n * Imperative DOM helpers for the contentEditable view. Each block is a\n * block-level element carrying `data-block-id`; its inline runs are spans\n * carrying `data-off` (their start offset in the block) so DOM positions map\n * back to {blockId, offset} and vice-versa.\n */\n\nexport const BLOCK_SEL = \"[data-block-id]\";\n\nexport function blockElOf(node: Node | null, root: HTMLElement): HTMLElement | null {\n let n: Node | null = node;\n while (n && n !== root) {\n if (n instanceof HTMLElement && n.dataset.blockId) return n;\n n = n.parentNode;\n }\n return null;\n}\n\nfunction spanOf(node: Node | null): HTMLElement | null {\n let n: Node | null = node;\n while (n) {\n if (n instanceof HTMLElement && n.dataset.off != null) return n;\n n = n.parentNode;\n }\n return null;\n}\n\n/** Map a DOM (node, offset) to a {blockId, offset} model position. */\nexport function domToModel(\n root: HTMLElement,\n node: Node | null,\n offset: number,\n): { blockId: string; offset: number } | null {\n const blockEl = blockElOf(node, root);\n if (!blockEl) return null;\n const blockId = blockEl.dataset.blockId as string;\n\n if (node && node.nodeType === Node.TEXT_NODE) {\n const span = spanOf(node);\n const base = span ? Number(span.dataset.off) : 0;\n return { blockId, offset: base + offset };\n }\n\n // node is an element; `offset` is a child index. Resolve via the child spans.\n const el = node as HTMLElement;\n if (el.dataset?.off != null) {\n // selection landed on a span boundary\n return { blockId, offset: Number(el.dataset.off) + (offset > 0 ? spanLen(el) : 0) };\n }\n const kids = Array.from(el.childNodes);\n for (let i = offset; i < kids.length; i++) {\n const k = kids[i];\n if (k instanceof HTMLElement && k.dataset.off != null) return { blockId, offset: Number(k.dataset.off) };\n }\n // past the last span → block end\n let end = 0;\n for (const k of kids) if (k instanceof HTMLElement && k.dataset.off != null) end = Math.max(end, Number(k.dataset.off) + spanLen(k));\n return { blockId, offset: end };\n}\n\nfunction spanLen(span: HTMLElement): number {\n return span.dataset.len != null ? Number(span.dataset.len) : (span.textContent ?? \"\").length;\n}\n\n/** Find the DOM (node, offset) for a {blockId, offset} model position. */\nexport function modelToDom(\n root: HTMLElement,\n blockId: string,\n offset: number,\n): { node: Node; offset: number } | null {\n const blockEl = root.querySelector(`[data-block-id=\"${esc(blockId)}\"]`) as HTMLElement | null;\n if (!blockEl) return null;\n const spans = Array.from(blockEl.querySelectorAll(\"[data-off]\")) as HTMLElement[];\n if (spans.length === 0) {\n return { node: blockEl, offset: 0 }; // empty block\n }\n for (const span of spans) {\n const start = Number(span.dataset.off);\n const len = spanLen(span);\n if (offset <= start + len) {\n if (span.dataset.atom != null) {\n // atom: place before or after it (offset is start or start+1)\n const idx = Array.prototype.indexOf.call(blockEl.childNodes, span);\n return { node: blockEl, offset: offset <= start ? idx : idx + 1 };\n }\n const textNode = span.firstChild ?? span;\n return { node: textNode, offset: Math.max(0, Math.min(offset - start, (textNode.textContent ?? \"\").length)) };\n }\n }\n // past everything → after the last span\n const last = spans[spans.length - 1];\n const textNode = last.firstChild ?? last;\n return { node: textNode, offset: (textNode.textContent ?? \"\").length };\n}\n\nfunction markClass(marks: InlineItem[\"marks\"]): string {\n const m = marks ?? {};\n const cls = [\"ori-frag\"];\n if (m.bold) cls.push(\"ori-m-bold\");\n if (m.italic) cls.push(\"ori-m-italic\");\n if (m.underline) cls.push(\"ori-m-underline\");\n if (m.strike) cls.push(\"ori-m-strike\");\n if (m.code) cls.push(\"ori-frag-code\");\n if (m.link) cls.push(\"ori-frag-link\");\n return cls.join(\" \");\n}\n\n/** Build the inline run DOM for a block (text spans only; atoms handled by the view). */\nexport function buildRun(item: InlineItem): HTMLElement {\n const span = document.createElement(\"span\");\n span.className = markClass(item.marks);\n span.dataset.off = String(item.start);\n span.dataset.len = String(item.text.length);\n span.textContent = item.text;\n return span;\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport { isCollapsed } from \"@wingleeio/ori-core\";\nimport type { ReactNode } from \"react\";\nimport { createRoot, type Root } from \"react-dom/client\";\nimport type { AtomRenderer, BlockRenderer } from \"../renderers\";\nimport { blockElOf, buildRun, domToModel, esc, modelToDom } from \"./dom\";\n\nconst PLACEHOLDER = \"\";\n\nexport interface ViewOptions {\n readOnly?: boolean;\n renderAtom: (type: string) => AtomRenderer | undefined;\n renderBlock: (type: string) => BlockRenderer | undefined;\n}\n\n/**\n * Imperative contentEditable view over an {@link EditorController}. The browser\n * owns caret / selection / trackpad / menus / IME on the live text; we intercept\n * structural + cross-block edits (beforeinput) and route them through the\n * controller, let smooth in-block typing flow natively and read it back, and\n * keep the DOM selection and the controller selection in lock-step.\n */\nexport class EditorView {\n private roots = new Map<HTMLElement, Root>();\n private composing = false;\n private applyingModel = false;\n private detachers: Array<() => void> = [];\n /** The model revision the DOM currently reflects (so external changes — remote\n * edits, app commands — re-render, but our own edits don't clobber the caret). */\n private lastRevision = -1;\n\n constructor(\n private root: HTMLElement,\n private editor: EditorController,\n private opts: ViewOptions,\n ) {\n root.setAttribute(\"contenteditable\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"spellcheck\", opts.readOnly ? \"false\" : \"true\");\n root.setAttribute(\"role\", \"textbox\");\n root.setAttribute(\"aria-multiline\", \"true\");\n this.renderBlocks();\n this.lastRevision = this.rev();\n\n const on = <K extends keyof HTMLElementEventMap>(\n t: K,\n h: (e: HTMLElementEventMap[K]) => void,\n o?: AddEventListenerOptions,\n ) => {\n root.addEventListener(t, h as EventListener, o);\n this.detachers.push(() => root.removeEventListener(t, h as EventListener, o));\n };\n on(\"beforeinput\", (e) => this.onBeforeInput(e as InputEvent));\n on(\"input\", () => this.onInput());\n on(\"keydown\", (e) => this.onKeyDown(e as KeyboardEvent));\n on(\"blur\", () => {\n // Clicking outside the editor drops the selection (so a selection toolbar\n // hides). Defer so we can ignore a window/tab blur and focus-preserving\n // clicks (e.g. toolbar buttons that re-focus the editor).\n setTimeout(() => {\n if (document.activeElement === this.root || !document.hasFocus()) return;\n const sel = this.editor.getSelection();\n if (sel && !isCollapsed(sel)) {\n this.editor.collapse(sel.focus);\n this.lastRevision = this.rev();\n }\n }, 0);\n });\n on(\"compositionstart\", () => (this.composing = true));\n on(\"compositionend\", () => {\n this.composing = false;\n this.onInput();\n });\n\n const onSelChange = () => {\n if (this.applyingModel || this.composing) return;\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n // DOM is already the source of truth here — record the revision so the\n // resulting React sync() doesn't write the selection back and collapse it.\n this.lastRevision = this.rev();\n };\n document.addEventListener(\"selectionchange\", onSelChange);\n this.detachers.push(() => document.removeEventListener(\"selectionchange\", onSelChange));\n }\n\n destroy() {\n this.detachers.forEach((d) => d());\n this.roots.forEach((r) => r.unmount());\n this.roots.clear();\n }\n\n focus() {\n this.root.focus();\n }\n\n // --- rendering ---------------------------------------------------------\n\n private rev(): number {\n return this.editor.getSnapshot().revision;\n }\n\n /**\n * Called by React on every model change. Only re-renders when the model moved\n * ahead of what we last drew (an *external* change — app command, undo, remote);\n * our own edits already updated the DOM and must not be clobbered.\n */\n sync() {\n const rev = this.rev();\n if (rev === this.lastRevision) return;\n // Only restore the DOM selection when the *content* changed (app command,\n // undo, remote). A selection-only change is already correct in the DOM, and\n // writing it back would fight (and collapse) the user's native selection.\n const changed = this.renderBlocks();\n if (changed) this.writeSelection();\n this.lastRevision = rev;\n }\n\n /** After a controlled (preventDefault'd) edit: re-render + restore the caret. */\n private commit() {\n this.renderBlocks();\n this.writeSelection();\n this.lastRevision = this.rev();\n }\n\n /** A content signature for a block, so unchanged blocks aren't re-rendered. */\n private sig(id: string): string {\n return this.editor.getBlockType(id) + \"|\" + JSON.stringify(this.editor.getInline(id));\n }\n\n /**\n * Reconcile the DOM to the *visible window* of blocks (virtualization): a top\n * spacer, the windowed blocks, then a bottom spacer — heights from the\n * controller's offscreen measurement. On-screen blocks are reused by id so a\n * caret inside one survives a scroll. Returns true if the DOM was mutated.\n */\n private renderBlocks(): boolean {\n let changed = false;\n const snap = this.editor.getSnapshot();\n const vis = snap.visible;\n const topH = vis.length ? vis[0].top : 0;\n const botH = vis.length\n ? Math.max(0, snap.totalHeight - (vis[vis.length - 1].top + vis[vis.length - 1].height))\n : Math.max(0, snap.totalHeight);\n\n const TOP = \"\u0000top\";\n const BOTTOM = \"\u0000bottom\";\n const want = [TOP, ...vis.map((v) => v.id), BOTTOM];\n const keyOf = (el: Element) => {\n const e = el as HTMLElement;\n return e.dataset.spacer ? \"\u0000\" + e.dataset.spacer : (e.dataset.blockId ?? \"\");\n };\n const have = new Map<string, HTMLElement>();\n for (const c of Array.from(this.root.children)) have.set(keyOf(c), c as HTMLElement);\n\n let prev: HTMLElement | null = null;\n for (const k of want) {\n let el: HTMLElement | undefined = have.get(k);\n if (el) {\n have.delete(k);\n } else {\n el = k === TOP || k === BOTTOM ? this.makeSpacer(k.slice(1)) : this.makeBlock(k);\n changed = true;\n }\n const anchor: ChildNode | null = prev ? prev.nextSibling : this.root.firstChild;\n if (anchor !== el) {\n this.root.insertBefore(el, anchor);\n changed = true;\n }\n prev = el;\n }\n for (const el of have.values()) {\n if (el.dataset.blockId) this.unmountRootsIn(el);\n el.remove();\n changed = true;\n }\n\n const top = this.root.firstElementChild as HTMLElement | null;\n if (top && top.style.height !== `${topH}px`) top.style.height = `${topH}px`;\n const bot = this.root.lastElementChild as HTMLElement | null;\n if (bot && bot.style.height !== `${botH}px`) bot.style.height = `${botH}px`;\n\n for (const vb of vis) {\n const el = this.root.querySelector(`[data-block-id=\"${esc(vb.id)}\"]`) as HTMLElement | null;\n if (!el) continue;\n const sig = this.sig(vb.id);\n if (el.dataset.sig !== sig) {\n el.dataset.sig = sig;\n this.renderBlockInner(el, vb.id);\n changed = true;\n }\n }\n return changed;\n }\n\n private makeBlock(id: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.blockId = id;\n return el;\n }\n\n private makeSpacer(which: string): HTMLElement {\n const el = document.createElement(\"div\");\n el.dataset.spacer = which;\n el.setAttribute(\"contenteditable\", \"false\");\n el.setAttribute(\"aria-hidden\", \"true\");\n el.style.userSelect = \"none\";\n el.style.pointerEvents = \"none\";\n return el;\n }\n\n private renderBlockInner(el: HTMLElement, id: string) {\n this.unmountRootsIn(el);\n const type = this.editor.getBlockType(id);\n el.className = `ori-block ori-block-${type}`;\n\n const blockRenderer = this.opts.renderBlock(type);\n if (blockRenderer) {\n el.contentEditable = \"false\";\n el.textContent = \"\";\n const root = createRoot(el);\n root.render(blockRenderer({ editor: this.editor, block: { id, type, index: 0, top: 0, height: 0 }, layout: this.editor.getLayout(id)! }) as ReactNode);\n this.roots.set(el, root);\n return;\n }\n el.contentEditable = \"inherit\";\n el.textContent = \"\";\n const items = this.editor.getInline(id);\n if (items.length === 0) {\n el.appendChild(document.createElement(\"br\")); // keep an empty block selectable\n return;\n }\n for (const item of items) {\n if (item.atom) {\n const span = document.createElement(\"span\");\n span.className = \"ori-atom\";\n span.contentEditable = \"false\";\n span.dataset.atom = \"true\";\n span.dataset.off = String(item.start);\n span.dataset.len = \"1\";\n el.appendChild(span);\n const renderer = this.opts.renderAtom(item.atom.type);\n if (renderer) {\n const r = createRoot(span);\n r.render(renderer({ editor: this.editor, atom: item.atom }) as ReactNode);\n this.roots.set(span, r);\n }\n } else {\n el.appendChild(buildRun(item));\n }\n }\n }\n\n private unmountRootsIn(el: HTMLElement) {\n for (const [node, root] of this.roots) {\n if (el === node || el.contains(node)) {\n root.unmount();\n this.roots.delete(node);\n }\n }\n }\n\n // --- selection ---------------------------------------------------------\n\n private readSelection() {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || !this.root.contains(s.anchorNode)) return null;\n const a = domToModel(this.root, s.anchorNode, s.anchorOffset);\n const f = domToModel(this.root, s.focusNode, s.focusOffset);\n if (!a || !f) return null;\n return { anchor: { blockId: a.blockId, offset: a.offset }, focus: { blockId: f.blockId, offset: f.offset } };\n }\n\n /** Push the controller's selection back into the DOM (after a model op). */\n private writeSelection() {\n const sel = this.editor.getSelection();\n if (!sel) return;\n const a = modelToDom(this.root, sel.anchor.blockId, sel.anchor.offset);\n const f = modelToDom(this.root, sel.focus.blockId, sel.focus.offset);\n if (!a || !f) return;\n const r = document.createRange();\n const s = window.getSelection();\n if (!s) return;\n this.applyingModel = true;\n try {\n r.setStart(a.node, a.offset);\n s.removeAllRanges();\n s.addRange(r);\n s.extend(f.node, f.offset);\n } catch {\n /* node detached mid-reconcile */\n } finally {\n this.applyingModel = false;\n }\n }\n\n /** The block text as the model sees it (atoms collapse to one placeholder). */\n private domBlockText(el: HTMLElement): string {\n let out = \"\";\n for (const child of Array.from(el.childNodes)) {\n if (child instanceof HTMLElement && child.dataset.atom != null) {\n out += PLACEHOLDER;\n } else {\n out += child.textContent ?? \"\";\n }\n }\n return out;\n }\n\n // --- input -------------------------------------------------------------\n\n /** Formatting + history shortcuts (the browser fires these as keydown). */\n private onKeyDown(e: KeyboardEvent) {\n if (this.opts.readOnly) return;\n const mod = e.metaKey || e.ctrlKey;\n if (!mod || e.altKey) return;\n const k = e.key.toLowerCase();\n const mark = ({ b: \"bold\", i: \"italic\", u: \"underline\", e: \"code\" } as const)[k];\n if (mark) {\n e.preventDefault();\n const sel = this.readSelection();\n if (sel) this.editor.setSelection(sel);\n this.editor.toggleMark(mark);\n this.commit();\n } else if (k === \"z\") {\n e.preventDefault();\n if (e.shiftKey) this.editor.redo();\n else this.editor.undo();\n this.commit();\n } else if (k === \"y\") {\n e.preventDefault();\n this.editor.redo();\n this.commit();\n }\n }\n\n private onBeforeInput(e: InputEvent) {\n if (this.opts.readOnly) {\n e.preventDefault();\n return;\n }\n const sel = this.readSelection();\n if (!sel) return;\n this.editor.setSelection(sel);\n const collapsed = isCollapsed(sel);\n const startOffset = this.editor.orderedSelection()?.start.offset ?? sel.focus.offset;\n const t = e.inputType;\n\n // Native fast path: collapsed in-block typing / deletion. The browser mutates\n // a single text node; onInput reads it back. Keeps autocorrect/IME native.\n if (collapsed && (t === \"insertText\" || t === \"insertCompositionText\" || t === \"insertReplacementText\")) return;\n if (collapsed && t === \"deleteContentForward\") return;\n if (collapsed && t === \"deleteContentBackward\" && startOffset > 0) return;\n\n // Everything else (structural + cross-block) is handled through the controller.\n const ed = this.editor;\n if (t === \"insertParagraph\") {\n e.preventDefault();\n ed.insertParagraphBreak();\n } else if (t.startsWith(\"delete\")) {\n e.preventDefault();\n if (t === \"deleteContentForward\") ed.deleteForward();\n else ed.deleteBackward();\n } else if (t === \"insertText\" || t === \"insertReplacementText\" || t === \"insertFromPaste\") {\n e.preventDefault();\n const text = e.data ?? e.dataTransfer?.getData(\"text/plain\") ?? \"\";\n if (text) ed.insertText(text);\n } else if (t === \"insertLineBreak\") {\n e.preventDefault();\n ed.insertText(\"\\n\");\n } else {\n return; // let the browser handle anything we don't model\n }\n this.commit();\n }\n\n private onInput() {\n if (this.composing || this.opts.readOnly) return;\n const blockEl = blockElOf(window.getSelection()?.anchorNode ?? null, this.root);\n if (!blockEl) {\n // structure changed under us (browser merged blocks) → full resync\n this.renderBlocks();\n this.lastRevision = this.rev();\n return;\n }\n const id = blockEl.dataset.blockId as string;\n const next = this.domBlockText(blockEl);\n const cur = this.editor.getBlockText(id);\n if (next === cur) return;\n\n // diff → splice through the controller (which infers marks at the caret)\n const max = Math.min(cur.length, next.length);\n let p = 0;\n while (p < max && cur[p] === next[p]) p++;\n let s = 0;\n while (s < max - p && cur[cur.length - 1 - s] === next[next.length - 1 - s]) s++;\n const from = p;\n const to = cur.length - s;\n const insert = next.slice(p, next.length - s);\n this.editor.setSelection({ anchor: { blockId: id, offset: from }, focus: { blockId: id, offset: to } });\n if (to > from) this.editor.deleteBackward();\n if (insert) this.editor.insertText(insert);\n // The browser already painted the text; just realign the run offsets.\n this.reindex(blockEl);\n this.lastRevision = this.rev();\n }\n\n /** Re-derive data-off / data-len after a native edit (no node replacement). */\n private reindex(el: HTMLElement) {\n let off = 0;\n for (const child of Array.from(el.children) as HTMLElement[]) {\n if (child.dataset.off == null) continue;\n child.dataset.off = String(off);\n const len = child.dataset.atom != null ? 1 : (child.textContent ?? \"\").length;\n child.dataset.len = String(len);\n off += len;\n }\n }\n}\n","import type { EditorController } from \"@wingleeio/ori-core\";\nimport {\n forwardRef,\n useEffect,\n useImperativeHandle,\n useLayoutEffect,\n useRef,\n useState,\n type CSSProperties,\n type PointerEvent as ReactPointerEvent,\n} from \"react\";\nimport { EditorView } from \"./ce/view\";\nimport { useEditorSnapshot } from \"./hooks\";\nimport type { AtomRenderer, BlockRenderer } from \"./renderers\";\n\nexport interface NoteEditorProps {\n editor: EditorController;\n className?: string;\n style?: CSSProperties;\n /** Max width of the centered content column, in px. */\n maxWidth?: number;\n placeholder?: string;\n autoFocus?: boolean;\n readOnly?: boolean;\n /** Renderers for custom (atomic) block node types. */\n blockRenderers?: Record<string, BlockRenderer>;\n /** Renderers for custom inline atom types. */\n atomRenderers?: Record<string, AtomRenderer>;\n}\n\n/** A viewport-space rectangle (client coordinates). */\nexport interface ViewportRect {\n top: number;\n left: number;\n right: number;\n bottom: number;\n width: number;\n height: number;\n}\n\n/** Imperative handle for building floating UI (slash / selection menus). */\nexport interface NoteEditorHandle {\n focus(): void;\n /** Caret position in viewport coordinates, or null if unavailable. */\n getCaretRect(): { x: number; y: number; height: number } | null;\n /** Bounding box of the current selection in viewport coordinates, or null. */\n getSelectionRect(): ViewportRect | null;\n /** The scrolling element, for scroll-aware positioning. */\n getScrollElement(): HTMLElement | null;\n /**\n * The content overlay element (a positioned layer that scrolls *with* the\n * text). Render floating UI into this with `position: absolute` and\n * content-relative coordinates so it rides the scroll natively instead of\n * trailing it (which causes a fixed-position toolbar to shake on scroll).\n */\n getOverlayElement(): HTMLElement | null;\n}\n\nfunction caretClientRect(): DOMRect | null {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0) return null;\n const r = s.getRangeAt(0).cloneRange();\n r.collapse(s.focusNode === r.endContainer && s.focusOffset === r.endOffset ? false : true);\n const rects = r.getClientRects();\n if (rects.length) return rects[rects.length - 1];\n const b = r.getBoundingClientRect();\n if (b.height || b.width) return b;\n // Empty block (`<br>` only): a collapsed range there has no client rects, so\n // synthesize the caret from the block box + its line metrics. Without this the\n // custom caret would vanish on empty lines (the native caret is hidden).\n const node = r.startContainer;\n const el = (node.nodeType === Node.TEXT_NODE ? node.parentElement : (node as HTMLElement)) ?? null;\n if (!el) return null;\n const eb = el.getBoundingClientRect();\n const cs = getComputedStyle(el);\n const lh = parseFloat(cs.lineHeight) || parseFloat(cs.fontSize) * 1.4 || 18;\n const padL = parseFloat(cs.paddingLeft) || 0;\n const padT = parseFloat(cs.paddingTop) || 0;\n return new DOMRect(eb.left + padL, eb.top + padT, 0, lh);\n}\n\n/**\n * A contentEditable note editor: the browser owns caret, selection, trackpad,\n * native menus and IME on the live text, while edits are routed through the\n * {@link EditorController} (Y.Doc). A custom caret is drawn on top so it can be\n * branded/animated independently of the (hidden) native one.\n */\nexport const NoteEditor = forwardRef<NoteEditorHandle, NoteEditorProps>(function NoteEditor(\n { editor, className, style, maxWidth = 720, placeholder, autoFocus, readOnly, blockRenderers, atomRenderers },\n ref,\n) {\n const snapshot = useEditorSnapshot(editor);\n const scrollerRef = useRef<HTMLDivElement>(null);\n const overlayRef = useRef<HTMLDivElement>(null);\n const contentRef = useRef<HTMLDivElement>(null);\n const viewRef = useRef<EditorView | null>(null);\n const [focused, setFocused] = useState(false);\n const [caret, setCaret] = useState<{ x: number; y: number; h: number } | null>(null);\n\n // Keep the latest renderers reachable without recreating the view.\n const renderersRef = useRef({ blockRenderers, atomRenderers });\n renderersRef.current = { blockRenderers, atomRenderers };\n\n useImperativeHandle(\n ref,\n (): NoteEditorHandle => ({\n focus: () => contentRef.current?.focus(),\n getCaretRect: () => {\n const r = caretClientRect();\n return r ? { x: r.left, y: r.top, height: r.height || 16 } : null;\n },\n getSelectionRect: () => {\n const s = window.getSelection();\n if (!s || s.rangeCount === 0 || s.isCollapsed) return null;\n const b = s.getRangeAt(0).getBoundingClientRect();\n if (!b.width && !b.height) return null;\n return { top: b.top, left: b.left, right: b.right, bottom: b.bottom, width: b.width, height: b.height };\n },\n getScrollElement: () => scrollerRef.current,\n getOverlayElement: () => overlayRef.current,\n }),\n [],\n );\n\n // Create the imperative contentEditable view once.\n useEffect(() => {\n const el = contentRef.current;\n if (!el) return;\n const view = new EditorView(el, editor, {\n readOnly,\n renderAtom: (t) => renderersRef.current.atomRenderers?.[t],\n renderBlock: (t) => renderersRef.current.blockRenderers?.[t],\n });\n viewRef.current = view;\n return () => {\n view.destroy();\n viewRef.current = null;\n };\n }, [editor, readOnly]);\n\n // Reconcile the view when the model changes externally (app commands, undo,\n // remote). The view ignores revisions it produced itself (native typing).\n useEffect(() => {\n viewRef.current?.sync();\n }, [snapshot.revision]);\n\n useEffect(() => {\n if (autoFocus) contentRef.current?.focus();\n }, [autoFocus]);\n\n // Position the custom caret from the live DOM selection.\n useEffect(() => {\n const update = () => {\n const content = contentRef.current;\n const s = window.getSelection();\n if (!content || !s || s.rangeCount === 0 || !s.isCollapsed || !content.contains(s.anchorNode)) {\n setCaret(null);\n return;\n }\n const r = caretClientRect();\n const box = content.getBoundingClientRect();\n if (r) setCaret({ x: r.left - box.left, y: r.top - box.top, h: r.height || 18 });\n };\n document.addEventListener(\"selectionchange\", update);\n const ro = new ResizeObserver(update);\n if (contentRef.current) ro.observe(contentRef.current);\n update();\n return () => {\n document.removeEventListener(\"selectionchange\", update);\n ro.disconnect();\n };\n }, []);\n\n // Drive virtualization: keep the controller's width + viewport in sync.\n useLayoutEffect(() => {\n const sc = scrollerRef.current;\n const content = contentRef.current;\n if (!sc || !content) return;\n const sync = () => {\n editor.setWidth(content.clientWidth);\n editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n sync();\n const ro = new ResizeObserver(sync);\n ro.observe(sc);\n ro.observe(content);\n return () => ro.disconnect();\n }, [editor]);\n\n const onScroll = () => {\n const sc = scrollerRef.current;\n if (sc) editor.setViewport(sc.scrollTop, sc.clientHeight);\n };\n\n // Clicking the empty region below the content should focus the editor and put\n // the caret at the document end (the usual \"click to keep writing\" affordance).\n const onPointerDown = (e: ReactPointerEvent<HTMLDivElement>) => {\n if (readOnly) return;\n // Only act on the editor's own empty surface — not on blocks (the browser\n // places the caret) and not on floating UI (menus) layered into the overlay.\n const t = e.target as HTMLElement;\n const onSurface =\n t === scrollerRef.current ||\n t === overlayRef.current ||\n t.classList.contains(\"ori-ce\") ||\n t.dataset.spacer != null;\n if (!onSurface) return;\n const content = contentRef.current;\n const blocks = content?.querySelectorAll(\"[data-block-id]\");\n const last = blocks && (blocks[blocks.length - 1] as HTMLElement | undefined);\n if (!content || !last) return;\n if (e.clientY <= last.getBoundingClientRect().bottom) return; // beside text, not below\n e.preventDefault();\n content.focus();\n const sel = window.getSelection();\n if (!sel) return;\n const range = document.createRange();\n range.selectNodeContents(last);\n range.collapse(false);\n sel.removeAllRanges();\n sel.addRange(range);\n };\n\n const showCaret = focused && !!caret && !readOnly;\n\n return (\n <div className={`ori-root${className ? ` ${className}` : \"\"}`} style={style}>\n <div className=\"ori-scroller\" ref={scrollerRef} onScroll={onScroll} onPointerDown={onPointerDown}>\n <div className=\"ori-content\" ref={overlayRef} style={{ maxWidth, marginInline: \"auto\", position: \"relative\" }}>\n <div\n className=\"ori-canvas ori-ce\"\n ref={contentRef}\n onFocus={() => setFocused(true)}\n onBlur={() => setFocused(false)}\n suppressContentEditableWarning\n />\n {showCaret && caret ? (\n <div\n className=\"ori-caret\"\n style={{ position: \"absolute\", left: caret.x, top: caret.y, height: caret.h, pointerEvents: \"none\" }}\n aria-hidden\n />\n ) : null}\n {snapshot.empty && placeholder ? (\n <div className=\"ori-placeholder\" aria-hidden>\n {placeholder}\n </div>\n ) : null}\n </div>\n </div>\n </div>\n );\n});\n","import type { BlockLayout, EditorController, InlineAtom, VisibleBlock } from \"@wingleeio/ori-core\";\nimport { createContext, useContext, type ReactNode } from \"react\";\n\n/** Props for a custom block renderer (atomic nodes: divider, image, …). */\nexport interface BlockRendererProps {\n editor: EditorController;\n block: VisibleBlock;\n /** The block's synthetic layout (atomic blocks: one line, no fragments). */\n layout: BlockLayout;\n}\nexport type BlockRenderer = (props: BlockRendererProps) => ReactNode;\n\n/** Props for a custom inline-atom renderer (mention chip, inline math, …). */\nexport interface AtomRendererProps {\n editor: EditorController;\n atom: InlineAtom;\n}\nexport type AtomRenderer = (props: AtomRendererProps) => ReactNode;\n\nexport interface Renderers {\n blocks: Record<string, BlockRenderer>;\n atoms: Record<string, AtomRenderer>;\n}\n\nconst EMPTY: Renderers = { blocks: {}, atoms: {} };\n\nconst RenderersContext = createContext<Renderers>(EMPTY);\nexport const RenderersProvider = RenderersContext.Provider;\nexport const useRenderers = (): Renderers => useContext(RenderersContext);\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@wingleeio/ori-react",
3
- "version": "0.1.0",
3
+ "version": "0.1.2",
4
4
  "description": "React bindings for the Ori virtualized, local-first note editor.",
5
5
  "type": "module",
6
6
  "license": "MIT",
package/styles.css CHANGED
@@ -225,28 +225,6 @@
225
225
  color: var(--ori-placeholder, rgba(120, 120, 120, 0.5));
226
226
  }
227
227
 
228
- /* Block-type accents — none of these shift text position. */
229
- .ori-block-quote {
230
- color: var(--ori-quote-text, rgba(90, 90, 90, 0.95));
231
- }
232
-
233
- .ori-block-quote::before {
234
- content: "";
235
- position: absolute;
236
- left: -16px;
237
- top: 1px;
238
- bottom: 1px;
239
- width: 3px;
240
- border-radius: 2px;
241
- background: var(--ori-quote-border, rgba(120, 120, 120, 0.5));
242
- }
243
-
244
- .ori-block-code {
245
- background: var(--ori-codeblock-bg, rgba(135, 131, 120, 0.1));
246
- border-radius: 8px;
247
- box-shadow: 0 0 0 10px var(--ori-codeblock-bg, rgba(135, 131, 120, 0.1));
248
- }
249
-
250
228
  .ori-input {
251
229
  font: inherit;
252
230
  }