@embedpdf/plugin-selection 1.0.17 → 1.0.19

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
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("@embedpdf/core"),t=require("@embedpdf/models"),i="selection",n={id:i,name:"Selection Plugin",version:"1.0.0",provides:["selection"],requires:["interaction-manager"],optional:[],defaultConfig:{enabled:!0}},o="CACHE_PAGE_GEOMETRY",r="SET_SELECTION",s="START_SELECTION",c="END_SELECTION",a="CLEAR_SELECTION",h="SET_RECTS",g="SET_SLICES",l="RESET";function d(e,t){return e.rects[t]??[]}function u(e,i){return t.boundingRect(d(e,i))}function p(e,t,i){if(!e||!t)return null;if(i<e.start.page||i>e.end.page)return null;const n=i===e.start.page?e.start.index:0,o=t.runs[t.runs.length-1],r=o.charStart+o.glyphs.length-1;return{from:n,to:i===e.end.page?e.end.index:r}}function x(e,t,i,n=!0){const o=[];for(const r of e.runs){const e=r.charStart,n=e+r.glyphs.length-1;if(n<t||e>i)continue;const s=Math.max(t,e)-e,c=Math.min(i,n)-e;let a=1/0,h=-1/0,g=1/0,l=-1/0,d=0;for(let t=s;t<=c;t++){const e=r.glyphs[t];2!==e.flags&&(a=Math.min(a,e.x),h=Math.max(h,e.x+e.width),g=Math.min(g,e.y),l=Math.max(l,e.y+e.height),d++)}a!==1/0&&d>0&&o.push({rect:{origin:{x:a,y:g},size:{width:h-a,height:l-g}},charCount:d})}return n?C(o):o.map((e=>e.rect))}function y(e,t){const i=Math.min(e.origin.x,t.origin.x),n=Math.min(e.origin.y,t.origin.y);return{origin:{x:i,y:n},size:{width:Math.max(e.origin.x+e.size.width,t.origin.x+t.size.width)-i,height:Math.max(e.origin.y+e.size.height,t.origin.y+t.size.height)-n}}}function m(e,t){const i=Math.max(e.origin.x,t.origin.x),n=Math.max(e.origin.y,t.origin.y),o=Math.min(e.origin.x+e.size.width,t.origin.x+t.size.width),r=Math.min(e.origin.y+e.size.height,t.origin.y+t.size.height);return{origin:{x:i,y:n},size:{width:Math.max(0,o-i),height:Math.max(0,r-n)}}}function f(e){return e.size.width<=0||e.size.height<=0}function S(e,t){if(f(e)||f(t))return 0;const i=y(e,t);if(i.size.height===e.size.height||i.size.height===t.size.height)return 1;return m(e,t).size.height/i.size.height}function E(e,t){const i=e.rect,n=t.rect;if(S(i,n)<.8)return!1;const o=1*i.size.width/e.charCount,r=1*n.size.width/t.charCount,s=i.origin.x-o,c=i.origin.x+i.size.width+o,a=n.origin.x-r;return s<n.origin.x+n.size.width+r&&c>a}function C(e){const t=[];let i=null,n=null;for(const o of e)i&&n?E(i,o)?n=y(n,o.rect):(t.push(n),n=o.rect):n=o.rect,i=o;return n&&!f(n)&&t.push(n),t}const T=class extends e.BasePlugin{constructor(t,i,n){super(t,i),this.engine=n,this.enabledModes=new Set(["pointerMode"]),this.selecting=!1,this.selChange$=e.createBehaviorEmitter(),this.textRetrieved$=e.createBehaviorEmitter(),this.copyToClipboard$=e.createEmitter(),this.beginSelection$=e.createEmitter(),this.endSelection$=e.createEmitter(),this.coreStore.onAction(e.SET_DOCUMENT,(e=>{this.dispatch({type:l})}))}async initialize(){}async destroy(){this.selChange$.clear()}buildCapability(){return{getGeometry:e=>this.getOrLoadGeometry(e),getFormattedSelection:()=>function(e){const t=[],i=Object.keys(e.rects).map(Number);for(const n of i){const i=e.rects[n]||[];if(0===i.length)continue;const o=u(e,n);o&&t.push({pageIndex:n,rect:o,segmentRects:i})}return t}(this.state),getFormattedSelectionForPage:e=>function(e,t){const i=e.rects[t]||[];if(0===i.length)return null;const n=u(e,t);return n?{pageIndex:t,rect:n,segmentRects:i}:null}(this.state,e),getHighlightRectsForPage:e=>d(this.state,e),getHighlightRects:()=>this.state.rects,getBoundingRectForPage:e=>u(this.state,e),getBoundingRects:()=>function(e){const i=[],n=e.rects;for(const o in n){const e=Number(o),r=t.boundingRect(n[e]);r&&i.push({page:e,rect:r})}return i}(this.state),begin:(e,t)=>this.beginSelection(e,t),update:(e,t)=>this.updateSelection(e,t),end:()=>this.endSelection(),clear:()=>this.clearSelection(),onCopyToClipboard:this.copyToClipboard$.on,onSelectionChange:this.selChange$.on,onTextRetrieved:this.textRetrieved$.on,onBeginSelection:this.beginSelection$.on,onEndSelection:this.endSelection$.on,getSelectedText:()=>this.getSelectedText(),copyToClipboard:()=>this.copyToClipboard(),enableForMode:e=>this.enabledModes.add(e),isEnabledForMode:e=>this.enabledModes.has(e),getState:()=>this.state}}getOrLoadGeometry(e){const i=this.state.geometry[e];if(i)return t.PdfTaskHelper.resolve(i);if(!this.coreState.core.document)return t.PdfTaskHelper.reject({code:t.PdfErrorCode.NotFound,message:"Doc Not Found"});const n=this.coreState.core.document.pages.find((t=>t.index===e)),r=this.engine.getPageGeometry(this.coreState.core.document,n);return r.wait((t=>{this.dispatch(((e,t)=>({type:o,payload:{page:e,geo:t}}))(e,t))}),t.ignore),r}beginSelection(e,t){this.selecting=!0,this.anchor={page:e,index:t},this.dispatch({type:s}),this.beginSelection$.emit({page:e,index:t})}endSelection(){this.selecting=!1,this.anchor=void 0,this.dispatch({type:c}),this.endSelection$.emit()}clearSelection(){this.selecting=!1,this.anchor=void 0,this.dispatch({type:a}),this.selChange$.emit(null)}updateSelection(e,t){if(!this.selecting||!this.anchor)return;const i=this.anchor,n=e>i.page||e===i.page&&t>=i.index,o={start:n?i:{page:e,index:t},end:n?{page:e,index:t}:i};this.dispatch({type:r,payload:o}),this.updateRectsAndSlices(o),this.selChange$.emit(o)}updateRectsAndSlices(e){const t={},i={};for(let n=e.start.page;n<=e.end.page;n++){const o=this.state.geometry[n],r=p(e,o,n);r&&(t[n]=x(o,r.from,r.to),i[n]={start:r.from,count:r.to-r.from+1})}this.dispatch((e=>({type:h,payload:e}))(t)),this.dispatch({type:g,payload:i})}getSelectedText(){if(!this.coreState.core.document||!this.state.selection)return t.PdfTaskHelper.reject({code:t.PdfErrorCode.NotFound,message:"Doc Not Found or No Selection"});const e=this.state.selection,i=[];for(let t=e.start.page;t<=e.end.page;t++){const e=this.state.slices[t];e&&i.push({pageIndex:t,charIndex:e.start,charCount:e.count})}if(0===i.length)return t.PdfTaskHelper.resolve([]);const n=this.engine.getTextSlices(this.coreState.core.document,i);return n.wait((e=>{this.textRetrieved$.emit(e)}),t.ignore),n}copyToClipboard(){this.getSelectedText().wait((e=>{this.copyToClipboard$.emit(e.join("\n"))}),t.ignore)}};T.id="selection";let b=T;const M={geometry:{},rects:{},slices:{},selection:null,active:!1,selecting:!1},z={manifest:n,create:(e,t)=>new b(i,e,t),reducer:(e=M,t)=>{switch(t.type){case o:{const{page:i,geo:n}=t.payload;return{...e,geometry:{...e.geometry,[i]:n}}}case r:return{...e,selection:t.payload,active:!0};case s:return{...e,selecting:!0,selection:null,rects:{}};case c:return{...e,selecting:!1};case a:return{...e,selecting:!1,selection:null,rects:{},active:!1};case h:return{...e,rects:t.payload};case g:return{...e,slices:t.payload};case l:return M;default:return e}},initialState:M};exports.SELECTION_PLUGIN_ID=i,exports.SelectionPlugin=b,exports.SelectionPluginPackage=z,exports.getVerticalOverlap=S,exports.glyphAt=function(e,t){for(const i of e.runs){if(!(t.y>=i.rect.y&&t.y<=i.rect.y+i.rect.height&&t.x>=i.rect.x&&t.x<=i.rect.x+i.rect.width))continue;const e=i.glyphs.findIndex((e=>t.x>=e.x&&t.x<=e.x+e.width&&t.y>=e.y&&t.y<=e.y+e.height));if(-1!==e)return i.charStart+e}return-1},exports.manifest=n,exports.mergeAdjacentRects=C,exports.rectIntersect=m,exports.rectIsEmpty=f,exports.rectUnion=y,exports.rectsWithinSlice=x,exports.shouldMergeHorizontalRects=E,exports.sliceBounds=p;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("@embedpdf/core"),t=require("@embedpdf/models"),i="selection",n={id:i,name:"Selection Plugin",version:"1.0.0",provides:["selection"],requires:["interaction-manager"],optional:[],defaultConfig:{enabled:!0}},r="CACHE_PAGE_GEOMETRY",o="SET_SELECTION",s="START_SELECTION",c="END_SELECTION",a="CLEAR_SELECTION",h="SET_RECTS",g="SET_SLICES",l="RESET";function d(e,t){return e.rects[t]??[]}function u(e,i){return t.boundingRect(d(e,i))}function p(e,t,i){if(!e||!t)return null;if(i<e.start.page||i>e.end.page)return null;const n=i===e.start.page?e.start.index:0,r=t.runs[t.runs.length-1],o=r.charStart+r.glyphs.length-1;return{from:n,to:i===e.end.page?e.end.index:o}}function y(e,t,i,n=!0){const r=[];for(const o of e.runs){const e=o.charStart,n=e+o.glyphs.length-1;if(n<t||e>i)continue;const s=Math.max(t,e)-e,c=Math.min(i,n)-e;let a=1/0,h=-1/0,g=1/0,l=-1/0,d=0;for(let t=s;t<=c;t++){const e=o.glyphs[t];2!==e.flags&&(a=Math.min(a,e.x),h=Math.max(h,e.x+e.width),g=Math.min(g,e.y),l=Math.max(l,e.y+e.height),d++)}a!==1/0&&d>0&&r.push({rect:{origin:{x:a,y:g},size:{width:h-a,height:l-g}},charCount:d})}return n?C(r):r.map((e=>e.rect))}function m(e,t){const i=Math.min(e.origin.x,t.origin.x),n=Math.min(e.origin.y,t.origin.y);return{origin:{x:i,y:n},size:{width:Math.max(e.origin.x+e.size.width,t.origin.x+t.size.width)-i,height:Math.max(e.origin.y+e.size.height,t.origin.y+t.size.height)-n}}}function x(e,t){const i=Math.max(e.origin.x,t.origin.x),n=Math.max(e.origin.y,t.origin.y),r=Math.min(e.origin.x+e.size.width,t.origin.x+t.size.width),o=Math.min(e.origin.y+e.size.height,t.origin.y+t.size.height);return{origin:{x:i,y:n},size:{width:Math.max(0,r-i),height:Math.max(0,o-n)}}}function f(e){return e.size.width<=0||e.size.height<=0}function S(e,t){if(f(e)||f(t))return 0;const i=m(e,t);if(i.size.height===e.size.height||i.size.height===t.size.height)return 1;return x(e,t).size.height/i.size.height}function E(e,t){const i=e.rect,n=t.rect;if(S(i,n)<.8)return!1;const r=1*i.size.width/e.charCount,o=1*n.size.width/t.charCount,s=i.origin.x-r,c=i.origin.x+i.size.width+r,a=n.origin.x-o;return s<n.origin.x+n.size.width+o&&c>a}function C(e){const t=[];let i=null,n=null;for(const r of e)i&&n?E(i,r)?n=m(n,r.rect):(t.push(n),n=r.rect):n=r.rect,i=r;return n&&!f(n)&&t.push(n),t}const T=class extends e.BasePlugin{constructor(i,n){super(i,n),this.enabledModes=new Set(["pointerMode"]),this.selecting=!1,this.selChange$=e.createBehaviorEmitter(),this.textRetrieved$=e.createBehaviorEmitter(),this.copyToClipboard$=e.createEmitter(),this.beginSelection$=e.createEmitter(),this.endSelection$=e.createEmitter(),this.refreshPages$=e.createEmitter(),this.coreStore.onAction(e.SET_DOCUMENT,(e=>{this.dispatch({type:l})})),this.coreStore.onAction(e.REFRESH_PAGES,(e=>{const i=e.payload.map((e=>this.getNewPageGeometryAndCache(e)));t.Task.all(i).wait((()=>{this.refreshPages$.emit(e.payload)}),t.ignore)}))}async initialize(){}async destroy(){this.selChange$.clear()}buildCapability(){return{getGeometry:e=>this.getOrLoadGeometry(e),getFormattedSelection:()=>function(e){const t=[],i=Object.keys(e.rects).map(Number);for(const n of i){const i=e.rects[n]||[];if(0===i.length)continue;const r=u(e,n);r&&t.push({pageIndex:n,rect:r,segmentRects:i})}return t}(this.state),getFormattedSelectionForPage:e=>function(e,t){const i=e.rects[t]||[];if(0===i.length)return null;const n=u(e,t);return n?{pageIndex:t,rect:n,segmentRects:i}:null}(this.state,e),getHighlightRectsForPage:e=>d(this.state,e),getHighlightRects:()=>this.state.rects,getBoundingRectForPage:e=>u(this.state,e),getBoundingRects:()=>function(e){const i=[],n=e.rects;for(const r in n){const e=Number(r),o=t.boundingRect(n[e]);o&&i.push({page:e,rect:o})}return i}(this.state),begin:(e,t)=>this.beginSelection(e,t),update:(e,t)=>this.updateSelection(e,t),end:()=>this.endSelection(),clear:()=>this.clearSelection(),onCopyToClipboard:this.copyToClipboard$.on,onSelectionChange:this.selChange$.on,onTextRetrieved:this.textRetrieved$.on,onBeginSelection:this.beginSelection$.on,onEndSelection:this.endSelection$.on,getSelectedText:()=>this.getSelectedText(),copyToClipboard:()=>this.copyToClipboard(),enableForMode:e=>this.enabledModes.add(e),isEnabledForMode:e=>this.enabledModes.has(e),getState:()=>this.state}}onRefreshPages(e){return this.refreshPages$.on(e)}getNewPageGeometryAndCache(e){if(!this.coreState.core.document)return t.PdfTaskHelper.reject({code:t.PdfErrorCode.NotFound,message:"Doc Not Found"});const i=this.coreState.core.document.pages.find((t=>t.index===e)),n=this.engine.getPageGeometry(this.coreState.core.document,i);return n.wait((t=>{this.dispatch(((e,t)=>({type:r,payload:{page:e,geo:t}}))(e,t))}),t.ignore),n}getOrLoadGeometry(e){const i=this.state.geometry[e];return i?t.PdfTaskHelper.resolve(i):this.getNewPageGeometryAndCache(e)}beginSelection(e,t){this.selecting=!0,this.anchor={page:e,index:t},this.dispatch({type:s}),this.beginSelection$.emit({page:e,index:t})}endSelection(){this.selecting=!1,this.anchor=void 0,this.dispatch({type:c}),this.endSelection$.emit()}clearSelection(){this.selecting=!1,this.anchor=void 0,this.dispatch({type:a}),this.selChange$.emit(null)}updateSelection(e,t){if(!this.selecting||!this.anchor)return;const i=this.anchor,n=e>i.page||e===i.page&&t>=i.index,r={start:n?i:{page:e,index:t},end:n?{page:e,index:t}:i};this.dispatch({type:o,payload:r}),this.updateRectsAndSlices(r),this.selChange$.emit(r)}updateRectsAndSlices(e){const t={},i={};for(let n=e.start.page;n<=e.end.page;n++){const r=this.state.geometry[n],o=p(e,r,n);o&&(t[n]=y(r,o.from,o.to),i[n]={start:o.from,count:o.to-o.from+1})}this.dispatch((e=>({type:h,payload:e}))(t)),this.dispatch({type:g,payload:i})}getSelectedText(){if(!this.coreState.core.document||!this.state.selection)return t.PdfTaskHelper.reject({code:t.PdfErrorCode.NotFound,message:"Doc Not Found or No Selection"});const e=this.state.selection,i=[];for(let t=e.start.page;t<=e.end.page;t++){const e=this.state.slices[t];e&&i.push({pageIndex:t,charIndex:e.start,charCount:e.count})}if(0===i.length)return t.PdfTaskHelper.resolve([]);const n=this.engine.getTextSlices(this.coreState.core.document,i);return n.wait((e=>{this.textRetrieved$.emit(e)}),t.ignore),n}copyToClipboard(){this.getSelectedText().wait((e=>{this.copyToClipboard$.emit(e.join("\n"))}),t.ignore)}};T.id="selection";let b=T;const M={geometry:{},rects:{},slices:{},selection:null,active:!1,selecting:!1},P={manifest:n,create:e=>new b(i,e),reducer:(e=M,t)=>{switch(t.type){case r:{const{page:i,geo:n}=t.payload;return{...e,geometry:{...e.geometry,[i]:n}}}case o:return{...e,selection:t.payload,active:!0};case s:return{...e,selecting:!0,selection:null,rects:{}};case c:return{...e,selecting:!1};case a:return{...e,selecting:!1,selection:null,rects:{},active:!1};case h:return{...e,rects:t.payload};case g:return{...e,slices:t.payload};case l:return M;default:return e}},initialState:M};exports.SELECTION_PLUGIN_ID=i,exports.SelectionPlugin=b,exports.SelectionPluginPackage=P,exports.getVerticalOverlap=S,exports.glyphAt=function(e,t){for(const i of e.runs){if(!(t.y>=i.rect.y&&t.y<=i.rect.y+i.rect.height&&t.x>=i.rect.x&&t.x<=i.rect.x+i.rect.width))continue;const e=i.glyphs.findIndex((e=>t.x>=e.x&&t.x<=e.x+e.width&&t.y>=e.y&&t.y<=e.y+e.height));if(-1!==e)return i.charStart+e}return-1},exports.manifest=n,exports.mergeAdjacentRects=C,exports.rectIntersect=x,exports.rectIsEmpty=f,exports.rectUnion=m,exports.rectsWithinSlice=y,exports.shouldMergeHorizontalRects=E,exports.sliceBounds=p;
2
2
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","sources":["../src/lib/manifest.ts","../src/lib/actions.ts","../src/lib/selectors.ts","../src/lib/utils.ts","../src/lib/selection-plugin.ts","../src/lib/reducer.ts","../src/lib/index.ts"],"sourcesContent":["import { PluginManifest } from '@embedpdf/core';\nimport { SelectionPluginConfig } from './types';\n\nexport const SELECTION_PLUGIN_ID = 'selection';\n\nexport const manifest: PluginManifest<SelectionPluginConfig> = {\n id: SELECTION_PLUGIN_ID,\n name: 'Selection Plugin',\n version: '1.0.0',\n provides: ['selection'],\n requires: ['interaction-manager'],\n optional: [],\n defaultConfig: {\n enabled: true,\n },\n};\n","import { Action } from '@embedpdf/core';\nimport { PdfPageGeometry, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\nexport const CACHE_PAGE_GEOMETRY = 'CACHE_PAGE_GEOMETRY';\nexport const SET_SELECTION = 'SET_SELECTION';\nexport const START_SELECTION = 'START_SELECTION';\nexport const END_SELECTION = 'END_SELECTION';\nexport const CLEAR_SELECTION = 'CLEAR_SELECTION';\nexport const SET_RECTS = 'SET_RECTS';\nexport const SET_SLICES = 'SET_SLICES';\nexport const RESET = 'RESET';\n\nexport interface CachePageGeometryAction extends Action {\n type: typeof CACHE_PAGE_GEOMETRY;\n payload: { page: number; geo: PdfPageGeometry };\n}\nexport interface SetSelectionAction extends Action {\n type: typeof SET_SELECTION;\n payload: SelectionRangeX | null;\n}\n\nexport interface StartSelectionAction extends Action {\n type: typeof START_SELECTION;\n}\n\nexport interface EndSelectionAction extends Action {\n type: typeof END_SELECTION;\n}\n\nexport interface ClearSelectionAction extends Action {\n type: typeof CLEAR_SELECTION;\n}\n\nexport interface SetRectsAction extends Action {\n type: typeof SET_RECTS;\n payload: Record<number, Rect[]>;\n}\n\nexport interface SetSlicesAction extends Action {\n type: typeof SET_SLICES;\n payload: Record<number, { start: number; count: number }>;\n}\n\nexport interface ResetAction extends Action {\n type: typeof RESET;\n}\n\nexport type SelectionAction =\n | CachePageGeometryAction\n | SetSelectionAction\n | StartSelectionAction\n | EndSelectionAction\n | ClearSelectionAction\n | SetRectsAction\n | SetSlicesAction\n | ResetAction;\n\nexport const cachePageGeometry = (page: number, geo: PdfPageGeometry): CachePageGeometryAction => ({\n type: CACHE_PAGE_GEOMETRY,\n payload: { page, geo },\n});\n\nexport const setSelection = (sel: SelectionRangeX): SetSelectionAction => ({\n type: SET_SELECTION,\n payload: sel,\n});\n\nexport const startSelection = (): StartSelectionAction => ({ type: START_SELECTION });\n\nexport const endSelection = (): EndSelectionAction => ({ type: END_SELECTION });\n\nexport const clearSelection = (): ClearSelectionAction => ({ type: CLEAR_SELECTION });\n\nexport const setRects = (allRects: Record<number, Rect[]>): SetRectsAction => ({\n type: SET_RECTS,\n payload: allRects,\n});\n\nexport const setSlices = (\n slices: Record<number, { start: number; count: number }>,\n): SetSlicesAction => ({ type: SET_SLICES, payload: slices });\n\nexport const reset = (): ResetAction => ({ type: RESET });\n","import { Rect, boundingRect } from '@embedpdf/models';\nimport { FormattedSelection, SelectionState } from './types';\n\nexport function selectRectsForPage(state: SelectionState, page: number) {\n return state.rects[page] ?? [];\n}\n\nexport function selectBoundingRectForPage(state: SelectionState, page: number) {\n return boundingRect(selectRectsForPage(state, page));\n}\n\nexport function selectRectsAndBoundingRectForPage(state: SelectionState, page: number) {\n return {\n rects: selectRectsForPage(state, page),\n boundingRect: selectBoundingRectForPage(state, page),\n };\n}\n\nexport function selectBoundingRectsForAllPages(state: SelectionState) {\n const out: { page: number; rect: Rect }[] = [];\n const rectMap = state.rects;\n\n for (const key in rectMap) {\n const page = Number(key);\n const bRect = boundingRect(rectMap[page]);\n if (bRect) out.push({ page, rect: bRect });\n }\n return out;\n}\n\nexport function getFormattedSelectionForPage(\n state: SelectionState,\n page: number,\n): FormattedSelection | null {\n const segmentRects = state.rects[page] || [];\n if (segmentRects.length === 0) return null;\n const boundingRect = selectBoundingRectForPage(state, page);\n if (!boundingRect) return null;\n return { pageIndex: page, rect: boundingRect, segmentRects };\n}\n\nexport function getFormattedSelection(state: SelectionState) {\n const result: FormattedSelection[] = [];\n\n // Get all pages that have rects\n const pages = Object.keys(state.rects).map(Number);\n\n for (const pageIndex of pages) {\n const segmentRects = state.rects[pageIndex] || [];\n\n if (segmentRects.length === 0) continue;\n\n // Calculate bounding rect for this page\n const boundingRect = selectBoundingRectForPage(state, pageIndex);\n\n if (boundingRect) {\n result.push({\n pageIndex,\n rect: boundingRect,\n segmentRects,\n });\n }\n }\n\n return result;\n}\n","import { PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\n/**\n * Hit-test helper using runs\n * @param geo - page geometry\n * @param pt - point\n * @returns glyph index\n */\nexport function glyphAt(geo: PdfPageGeometry, pt: Position) {\n for (const run of geo.runs) {\n const inRun =\n pt.y >= run.rect.y &&\n pt.y <= run.rect.y + run.rect.height &&\n pt.x >= run.rect.x &&\n pt.x <= run.rect.x + run.rect.width;\n\n if (!inRun) continue;\n\n // Simply check if the point is within any glyph's bounding box\n const rel = run.glyphs.findIndex(\n (g) => pt.x >= g.x && pt.x <= g.x + g.width && pt.y >= g.y && pt.y <= g.y + g.height,\n );\n\n if (rel !== -1) {\n return run.charStart + rel;\n }\n }\n return -1;\n}\n\n/**\n * Helper: min/max glyph indices on `page` for current sel\n * @param sel - selection range\n * @param geo - page geometry\n * @param page - page index\n * @returns { from: number; to: number } | null\n */\nexport function sliceBounds(\n sel: SelectionRangeX | null,\n geo: PdfPageGeometry | undefined,\n page: number,\n): { from: number; to: number } | null {\n if (!sel || !geo) return null;\n if (page < sel.start.page || page > sel.end.page) return null;\n\n const from = page === sel.start.page ? sel.start.index : 0;\n\n const lastRun = geo.runs[geo.runs.length - 1];\n const lastCharOnPage = lastRun.charStart + lastRun.glyphs.length - 1;\n\n const to = page === sel.end.page ? sel.end.index : lastCharOnPage;\n\n return { from, to };\n}\n\n/**\n * Helper: build rects for a slice of the page\n * @param geo - page geometry\n * @param from - from index\n * @param to - to index\n * @param merge - whether to merge adjacent rects (default: true)\n * @returns rects\n */\nexport function rectsWithinSlice(\n geo: PdfPageGeometry,\n from: number,\n to: number,\n merge: boolean = true,\n): Rect[] {\n const textRuns: TextRunInfo[] = [];\n\n for (const run of geo.runs) {\n const runStart = run.charStart;\n const runEnd = runStart + run.glyphs.length - 1;\n if (runEnd < from || runStart > to) continue;\n\n const sIdx = Math.max(from, runStart) - runStart;\n const eIdx = Math.min(to, runEnd) - runStart;\n\n let minX = Infinity,\n maxX = -Infinity;\n let minY = Infinity,\n maxY = -Infinity;\n let charCount = 0;\n\n for (let i = sIdx; i <= eIdx; i++) {\n const g = run.glyphs[i];\n if (g.flags === 2) continue; // empty glyph\n\n minX = Math.min(minX, g.x);\n maxX = Math.max(maxX, g.x + g.width);\n minY = Math.min(minY, g.y);\n maxY = Math.max(maxY, g.y + g.height);\n charCount++;\n }\n\n if (minX !== Infinity && charCount > 0) {\n textRuns.push({\n rect: {\n origin: { x: minX, y: minY },\n size: { width: maxX - minX, height: maxY - minY },\n },\n charCount,\n });\n }\n }\n\n // If merge is false, just return the individual rects\n if (!merge) {\n return textRuns.map((run) => run.rect);\n }\n\n // Otherwise merge adjacent rects\n return mergeAdjacentRects(textRuns);\n}\n\n/**\n * ============================================================================\n * Rectangle Merging Algorithm\n * ============================================================================\n *\n * The following code is adapted from Chromium's PDF text selection implementation.\n *\n * Copyright 2010 The Chromium Authors\n * Use of this source code is governed by a BSD-style license that can be\n * found in the LICENSE file: https://source.chromium.org/chromium/chromium/src/+/main:LICENSE\n *\n * Original source:\n * https://source.chromium.org/chromium/chromium/src/+/main:pdf/pdfium/pdfium_range.cc\n *\n * Adapted for TypeScript and this project's Rect/geometry types.\n */\n\n/**\n * Text run info for rect merging (similar to Chromium's ScreenRectTextRunInfo)\n */\nexport interface TextRunInfo {\n rect: Rect;\n charCount: number;\n}\n\n/**\n * Helper functions for Rect operations\n */\nexport function rectUnion(rect1: Rect, rect2: Rect): Rect {\n const left = Math.min(rect1.origin.x, rect2.origin.x);\n const top = Math.min(rect1.origin.y, rect2.origin.y);\n const right = Math.max(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.max(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n return {\n origin: { x: left, y: top },\n size: { width: right - left, height: bottom - top },\n };\n}\n\nexport function rectIntersect(rect1: Rect, rect2: Rect): Rect {\n const left = Math.max(rect1.origin.x, rect2.origin.x);\n const top = Math.max(rect1.origin.y, rect2.origin.y);\n const right = Math.min(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.min(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n const width = Math.max(0, right - left);\n const height = Math.max(0, bottom - top);\n\n return {\n origin: { x: left, y: top },\n size: { width, height },\n };\n}\n\nexport function rectIsEmpty(rect: Rect): boolean {\n return rect.size.width <= 0 || rect.size.height <= 0;\n}\n\n/**\n * Returns a ratio between [0, 1] representing vertical overlap\n */\nexport function getVerticalOverlap(rect1: Rect, rect2: Rect): number {\n if (rectIsEmpty(rect1) || rectIsEmpty(rect2)) return 0;\n\n const unionRect = rectUnion(rect1, rect2);\n\n if (unionRect.size.height === rect1.size.height || unionRect.size.height === rect2.size.height) {\n return 1.0;\n }\n\n const intersectRect = rectIntersect(rect1, rect2);\n return intersectRect.size.height / unionRect.size.height;\n}\n\n/**\n * Returns true if there is sufficient horizontal and vertical overlap\n */\nexport function shouldMergeHorizontalRects(textRun1: TextRunInfo, textRun2: TextRunInfo): boolean {\n const VERTICAL_OVERLAP_THRESHOLD = 0.8;\n const rect1 = textRun1.rect;\n const rect2 = textRun2.rect;\n\n if (getVerticalOverlap(rect1, rect2) < VERTICAL_OVERLAP_THRESHOLD) {\n return false;\n }\n\n const HORIZONTAL_WIDTH_FACTOR = 1.0;\n const averageWidth1 = (HORIZONTAL_WIDTH_FACTOR * rect1.size.width) / textRun1.charCount;\n const averageWidth2 = (HORIZONTAL_WIDTH_FACTOR * rect2.size.width) / textRun2.charCount;\n\n const rect1Left = rect1.origin.x - averageWidth1;\n const rect1Right = rect1.origin.x + rect1.size.width + averageWidth1;\n const rect2Left = rect2.origin.x - averageWidth2;\n const rect2Right = rect2.origin.x + rect2.size.width + averageWidth2;\n\n return rect1Left < rect2Right && rect1Right > rect2Left;\n}\n\n/**\n * Merge adjacent rectangles based on proximity and overlap (similar to Chromium's algorithm)\n */\nexport function mergeAdjacentRects(textRuns: TextRunInfo[]): Rect[] {\n const results: Rect[] = [];\n let previousTextRun: TextRunInfo | null = null;\n let currentRect: Rect | null = null;\n\n for (const textRun of textRuns) {\n if (previousTextRun && currentRect) {\n if (shouldMergeHorizontalRects(previousTextRun, textRun)) {\n currentRect = rectUnion(currentRect, textRun.rect);\n } else {\n results.push(currentRect);\n currentRect = textRun.rect;\n }\n } else {\n currentRect = textRun.rect;\n }\n previousTextRun = textRun;\n }\n\n if (currentRect && !rectIsEmpty(currentRect)) {\n results.push(currentRect);\n }\n\n return results;\n}\n","import {\n BasePlugin,\n PluginRegistry,\n SET_DOCUMENT,\n createBehaviorEmitter,\n createEmitter,\n} from '@embedpdf/core';\nimport {\n PdfEngine,\n PdfPageGeometry,\n Rect,\n PdfTask,\n PdfTaskHelper,\n PdfErrorCode,\n ignore,\n PageTextSlice,\n} from '@embedpdf/models';\n\nimport {\n cachePageGeometry,\n setSelection,\n SelectionAction,\n endSelection,\n startSelection,\n clearSelection,\n reset,\n setRects,\n setSlices,\n} from './actions';\nimport * as selector from './selectors';\nimport {\n SelectionCapability,\n SelectionPluginConfig,\n SelectionRangeX,\n SelectionState,\n} from './types';\nimport { sliceBounds, rectsWithinSlice } from './utils';\n\nexport class SelectionPlugin extends BasePlugin<\n SelectionPluginConfig,\n SelectionCapability,\n SelectionState,\n SelectionAction\n> {\n static readonly id = 'selection' as const;\n\n /** Modes that should trigger text-selection logic */\n private enabledModes = new Set<string>(['pointerMode']);\n\n /* interactive state */\n private selecting = false;\n private anchor?: { page: number; index: number };\n\n private readonly selChange$ = createBehaviorEmitter<SelectionState['selection']>();\n private readonly textRetrieved$ = createBehaviorEmitter<string[]>();\n private readonly copyToClipboard$ = createEmitter<string>();\n private readonly beginSelection$ = createEmitter<{ page: number; index: number }>();\n private readonly endSelection$ = createEmitter<void>();\n\n constructor(\n id: string,\n registry: PluginRegistry,\n private engine: PdfEngine,\n ) {\n super(id, registry);\n\n this.coreStore.onAction(SET_DOCUMENT, (_action) => {\n this.dispatch(reset());\n });\n }\n\n /* ── life-cycle ────────────────────────────────────────── */\n async initialize() {}\n async destroy() {\n this.selChange$.clear();\n }\n\n /* ── capability exposed to UI / other plugins ─────────── */\n buildCapability(): SelectionCapability {\n return {\n getGeometry: (p) => this.getOrLoadGeometry(p),\n getFormattedSelection: () => selector.getFormattedSelection(this.state),\n getFormattedSelectionForPage: (p) => selector.getFormattedSelectionForPage(this.state, p),\n getHighlightRectsForPage: (p) => selector.selectRectsForPage(this.state, p),\n getHighlightRects: () => this.state.rects,\n getBoundingRectForPage: (p) => selector.selectBoundingRectForPage(this.state, p),\n getBoundingRects: () => selector.selectBoundingRectsForAllPages(this.state),\n begin: (p, i) => this.beginSelection(p, i),\n update: (p, i) => this.updateSelection(p, i),\n end: () => this.endSelection(),\n clear: () => this.clearSelection(),\n onCopyToClipboard: this.copyToClipboard$.on,\n onSelectionChange: this.selChange$.on,\n onTextRetrieved: this.textRetrieved$.on,\n onBeginSelection: this.beginSelection$.on,\n onEndSelection: this.endSelection$.on,\n getSelectedText: () => this.getSelectedText(),\n copyToClipboard: () => this.copyToClipboard(),\n enableForMode: (id: string) => this.enabledModes.add(id),\n isEnabledForMode: (id: string) => this.enabledModes.has(id),\n getState: () => this.state,\n };\n }\n\n /* ── geometry cache ───────────────────────────────────── */\n private getOrLoadGeometry(pageIdx: number): PdfTask<PdfPageGeometry> {\n const cached = this.state.geometry[pageIdx];\n if (cached) return PdfTaskHelper.resolve(cached);\n\n if (!this.coreState.core.document)\n return PdfTaskHelper.reject({ code: PdfErrorCode.NotFound, message: 'Doc Not Found' });\n const page = this.coreState.core.document.pages.find((p) => p.index === pageIdx)!;\n\n const task = this.engine.getPageGeometry(this.coreState.core.document, page);\n\n task.wait((geo) => {\n this.dispatch(cachePageGeometry(pageIdx, geo));\n }, ignore);\n\n return task;\n }\n\n /* ── selection state updates ───────────────────────────── */\n private beginSelection(page: number, index: number) {\n this.selecting = true;\n this.anchor = { page, index };\n this.dispatch(startSelection());\n this.beginSelection$.emit({ page, index });\n }\n\n private endSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(endSelection());\n this.endSelection$.emit();\n }\n\n private clearSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(clearSelection());\n this.selChange$.emit(null);\n }\n\n private updateSelection(page: number, index: number) {\n if (!this.selecting || !this.anchor) return;\n\n const a = this.anchor;\n const forward = page > a.page || (page === a.page && index >= a.index);\n\n const start = forward ? a : { page, index };\n const end = forward ? { page, index } : a;\n\n const range = { start, end };\n this.dispatch(setSelection(range));\n this.updateRectsAndSlices(range);\n this.selChange$.emit(range);\n }\n\n private updateRectsAndSlices(range: SelectionRangeX) {\n const allRects: Record<number, Rect[]> = {};\n const allSlices: Record<number, { start: number; count: number }> = {};\n\n for (let p = range.start.page; p <= range.end.page; p++) {\n const geo = this.state.geometry[p];\n const sb = sliceBounds(range, geo, p);\n if (!sb) continue;\n\n allRects[p] = rectsWithinSlice(geo!, sb.from, sb.to);\n allSlices[p] = { start: sb.from, count: sb.to - sb.from + 1 };\n }\n\n this.dispatch(setRects(allRects));\n this.dispatch(setSlices(allSlices));\n }\n\n private getSelectedText(): PdfTask<string[]> {\n if (!this.coreState.core.document || !this.state.selection) {\n return PdfTaskHelper.reject({\n code: PdfErrorCode.NotFound,\n message: 'Doc Not Found or No Selection',\n });\n }\n\n const sel = this.state.selection;\n const req: PageTextSlice[] = [];\n\n for (let p = sel.start.page; p <= sel.end.page; p++) {\n const s = this.state.slices[p];\n if (s) req.push({ pageIndex: p, charIndex: s.start, charCount: s.count });\n }\n\n if (req.length === 0) return PdfTaskHelper.resolve([] as string[]);\n\n const task = this.engine.getTextSlices(this.coreState.core.document, req);\n\n // Emit the text when it's retrieved\n task.wait((text) => {\n this.textRetrieved$.emit(text);\n }, ignore);\n\n return task;\n }\n\n private copyToClipboard() {\n const text = this.getSelectedText();\n text.wait((text) => {\n this.copyToClipboard$.emit(text.join('\\n'));\n }, ignore);\n }\n}\n","import { SelectionState } from './types';\nimport {\n SelectionAction,\n CACHE_PAGE_GEOMETRY,\n SET_SELECTION,\n START_SELECTION,\n END_SELECTION,\n CLEAR_SELECTION,\n RESET,\n SET_SLICES,\n SET_RECTS,\n} from './actions';\n\nexport const initialState: SelectionState = {\n geometry: {},\n rects: {},\n slices: {},\n selection: null,\n active: false,\n selecting: false,\n};\n\nexport const selectionReducer = (state = initialState, action: SelectionAction): SelectionState => {\n switch (action.type) {\n case CACHE_PAGE_GEOMETRY: {\n const { page, geo } = action.payload;\n return { ...state, geometry: { ...state.geometry, [page]: geo } };\n }\n case SET_SELECTION:\n return { ...state, selection: action.payload, active: true };\n case START_SELECTION:\n return { ...state, selecting: true, selection: null, rects: {} };\n case END_SELECTION:\n return { ...state, selecting: false };\n case CLEAR_SELECTION:\n return { ...state, selecting: false, selection: null, rects: {}, active: false };\n case SET_RECTS:\n return { ...state, rects: action.payload };\n case SET_SLICES:\n return { ...state, slices: action.payload };\n case RESET:\n return initialState;\n default:\n return state;\n }\n};\n","import { PluginPackage } from '@embedpdf/core';\nimport { manifest, SELECTION_PLUGIN_ID } from './manifest';\nimport { SelectionPluginConfig, SelectionState } from './types';\n\nimport { SelectionPlugin } from './selection-plugin';\nimport { SelectionAction } from './actions';\nimport { selectionReducer, initialState } from './reducer';\n\nexport const SelectionPluginPackage: PluginPackage<\n SelectionPlugin,\n SelectionPluginConfig,\n SelectionState,\n SelectionAction\n> = {\n manifest,\n create: (registry, engine) => new SelectionPlugin(SELECTION_PLUGIN_ID, registry, engine),\n reducer: selectionReducer,\n initialState,\n};\n\nexport * from './selection-plugin';\nexport * from './types';\nexport * from './manifest';\nexport * from './utils';\n"],"names":["SELECTION_PLUGIN_ID","manifest","id","name","version","provides","requires","optional","defaultConfig","enabled","CACHE_PAGE_GEOMETRY","SET_SELECTION","START_SELECTION","END_SELECTION","CLEAR_SELECTION","SET_RECTS","SET_SLICES","RESET","selectRectsForPage","state","page","rects","selectBoundingRectForPage","boundingRect","sliceBounds","sel","geo","start","end","from","index","lastRun","runs","length","lastCharOnPage","charStart","glyphs","to","rectsWithinSlice","merge","textRuns","run","runStart","runEnd","sIdx","Math","max","eIdx","min","minX","Infinity","maxX","minY","maxY","charCount","i","g","flags","x","width","y","height","push","rect","origin","size","mergeAdjacentRects","map","rectUnion","rect1","rect2","left","top","rectIntersect","right","bottom","rectIsEmpty","getVerticalOverlap","unionRect","shouldMergeHorizontalRects","textRun1","textRun2","averageWidth1","averageWidth2","rect1Left","rect1Right","rect2Left","results","previousTextRun","currentRect","textRun","_SelectionPlugin","BasePlugin","constructor","registry","engine","super","this","enabledModes","Set","selecting","selChange$","createBehaviorEmitter","textRetrieved$","copyToClipboard$","createEmitter","beginSelection$","endSelection$","coreStore","onAction","SET_DOCUMENT","_action","dispatch","type","initialize","destroy","clear","buildCapability","getGeometry","p","getOrLoadGeometry","getFormattedSelection","result","pages","Object","keys","Number","pageIndex","segmentRects","selector.getFormattedSelection","getFormattedSelectionForPage","selector.getFormattedSelectionForPage","getHighlightRectsForPage","selector.selectRectsForPage","getHighlightRects","getBoundingRectForPage","selector.selectBoundingRectForPage","getBoundingRects","out","rectMap","key","bRect","selector.selectBoundingRectsForAllPages","begin","beginSelection","update","updateSelection","endSelection","clearSelection","onCopyToClipboard","on","onSelectionChange","onTextRetrieved","onBeginSelection","onEndSelection","getSelectedText","copyToClipboard","enableForMode","add","isEnabledForMode","has","getState","pageIdx","cached","geometry","PdfTaskHelper","resolve","coreState","core","document","reject","code","PdfErrorCode","NotFound","message","find","task","getPageGeometry","wait","payload","cachePageGeometry","ignore","anchor","emit","a","forward","range","updateRectsAndSlices","allRects","allSlices","sb","count","setRects","selection","req","s","slices","charIndex","getTextSlices","text","join","SelectionPlugin","initialState","active","SelectionPluginPackage","create","reducer","action","pt","rel","findIndex"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","sources":["../src/lib/manifest.ts","../src/lib/actions.ts","../src/lib/selectors.ts","../src/lib/utils.ts","../src/lib/selection-plugin.ts","../src/lib/reducer.ts","../src/lib/index.ts"],"sourcesContent":["import { PluginManifest } from '@embedpdf/core';\nimport { SelectionPluginConfig } from './types';\n\nexport const SELECTION_PLUGIN_ID = 'selection';\n\nexport const manifest: PluginManifest<SelectionPluginConfig> = {\n id: SELECTION_PLUGIN_ID,\n name: 'Selection Plugin',\n version: '1.0.0',\n provides: ['selection'],\n requires: ['interaction-manager'],\n optional: [],\n defaultConfig: {\n enabled: true,\n },\n};\n","import { Action } from '@embedpdf/core';\nimport { PdfPageGeometry, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\nexport const CACHE_PAGE_GEOMETRY = 'CACHE_PAGE_GEOMETRY';\nexport const SET_SELECTION = 'SET_SELECTION';\nexport const START_SELECTION = 'START_SELECTION';\nexport const END_SELECTION = 'END_SELECTION';\nexport const CLEAR_SELECTION = 'CLEAR_SELECTION';\nexport const SET_RECTS = 'SET_RECTS';\nexport const SET_SLICES = 'SET_SLICES';\nexport const RESET = 'RESET';\n\nexport interface CachePageGeometryAction extends Action {\n type: typeof CACHE_PAGE_GEOMETRY;\n payload: { page: number; geo: PdfPageGeometry };\n}\nexport interface SetSelectionAction extends Action {\n type: typeof SET_SELECTION;\n payload: SelectionRangeX | null;\n}\n\nexport interface StartSelectionAction extends Action {\n type: typeof START_SELECTION;\n}\n\nexport interface EndSelectionAction extends Action {\n type: typeof END_SELECTION;\n}\n\nexport interface ClearSelectionAction extends Action {\n type: typeof CLEAR_SELECTION;\n}\n\nexport interface SetRectsAction extends Action {\n type: typeof SET_RECTS;\n payload: Record<number, Rect[]>;\n}\n\nexport interface SetSlicesAction extends Action {\n type: typeof SET_SLICES;\n payload: Record<number, { start: number; count: number }>;\n}\n\nexport interface ResetAction extends Action {\n type: typeof RESET;\n}\n\nexport type SelectionAction =\n | CachePageGeometryAction\n | SetSelectionAction\n | StartSelectionAction\n | EndSelectionAction\n | ClearSelectionAction\n | SetRectsAction\n | SetSlicesAction\n | ResetAction;\n\nexport const cachePageGeometry = (page: number, geo: PdfPageGeometry): CachePageGeometryAction => ({\n type: CACHE_PAGE_GEOMETRY,\n payload: { page, geo },\n});\n\nexport const setSelection = (sel: SelectionRangeX): SetSelectionAction => ({\n type: SET_SELECTION,\n payload: sel,\n});\n\nexport const startSelection = (): StartSelectionAction => ({ type: START_SELECTION });\n\nexport const endSelection = (): EndSelectionAction => ({ type: END_SELECTION });\n\nexport const clearSelection = (): ClearSelectionAction => ({ type: CLEAR_SELECTION });\n\nexport const setRects = (allRects: Record<number, Rect[]>): SetRectsAction => ({\n type: SET_RECTS,\n payload: allRects,\n});\n\nexport const setSlices = (\n slices: Record<number, { start: number; count: number }>,\n): SetSlicesAction => ({ type: SET_SLICES, payload: slices });\n\nexport const reset = (): ResetAction => ({ type: RESET });\n","import { Rect, boundingRect } from '@embedpdf/models';\nimport { FormattedSelection, SelectionState } from './types';\n\nexport function selectRectsForPage(state: SelectionState, page: number) {\n return state.rects[page] ?? [];\n}\n\nexport function selectBoundingRectForPage(state: SelectionState, page: number) {\n return boundingRect(selectRectsForPage(state, page));\n}\n\nexport function selectRectsAndBoundingRectForPage(state: SelectionState, page: number) {\n return {\n rects: selectRectsForPage(state, page),\n boundingRect: selectBoundingRectForPage(state, page),\n };\n}\n\nexport function selectBoundingRectsForAllPages(state: SelectionState) {\n const out: { page: number; rect: Rect }[] = [];\n const rectMap = state.rects;\n\n for (const key in rectMap) {\n const page = Number(key);\n const bRect = boundingRect(rectMap[page]);\n if (bRect) out.push({ page, rect: bRect });\n }\n return out;\n}\n\nexport function getFormattedSelectionForPage(\n state: SelectionState,\n page: number,\n): FormattedSelection | null {\n const segmentRects = state.rects[page] || [];\n if (segmentRects.length === 0) return null;\n const boundingRect = selectBoundingRectForPage(state, page);\n if (!boundingRect) return null;\n return { pageIndex: page, rect: boundingRect, segmentRects };\n}\n\nexport function getFormattedSelection(state: SelectionState) {\n const result: FormattedSelection[] = [];\n\n // Get all pages that have rects\n const pages = Object.keys(state.rects).map(Number);\n\n for (const pageIndex of pages) {\n const segmentRects = state.rects[pageIndex] || [];\n\n if (segmentRects.length === 0) continue;\n\n // Calculate bounding rect for this page\n const boundingRect = selectBoundingRectForPage(state, pageIndex);\n\n if (boundingRect) {\n result.push({\n pageIndex,\n rect: boundingRect,\n segmentRects,\n });\n }\n }\n\n return result;\n}\n","import { PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\n/**\n * Hit-test helper using runs\n * @param geo - page geometry\n * @param pt - point\n * @returns glyph index\n */\nexport function glyphAt(geo: PdfPageGeometry, pt: Position) {\n for (const run of geo.runs) {\n const inRun =\n pt.y >= run.rect.y &&\n pt.y <= run.rect.y + run.rect.height &&\n pt.x >= run.rect.x &&\n pt.x <= run.rect.x + run.rect.width;\n\n if (!inRun) continue;\n\n // Simply check if the point is within any glyph's bounding box\n const rel = run.glyphs.findIndex(\n (g) => pt.x >= g.x && pt.x <= g.x + g.width && pt.y >= g.y && pt.y <= g.y + g.height,\n );\n\n if (rel !== -1) {\n return run.charStart + rel;\n }\n }\n return -1;\n}\n\n/**\n * Helper: min/max glyph indices on `page` for current sel\n * @param sel - selection range\n * @param geo - page geometry\n * @param page - page index\n * @returns { from: number; to: number } | null\n */\nexport function sliceBounds(\n sel: SelectionRangeX | null,\n geo: PdfPageGeometry | undefined,\n page: number,\n): { from: number; to: number } | null {\n if (!sel || !geo) return null;\n if (page < sel.start.page || page > sel.end.page) return null;\n\n const from = page === sel.start.page ? sel.start.index : 0;\n\n const lastRun = geo.runs[geo.runs.length - 1];\n const lastCharOnPage = lastRun.charStart + lastRun.glyphs.length - 1;\n\n const to = page === sel.end.page ? sel.end.index : lastCharOnPage;\n\n return { from, to };\n}\n\n/**\n * Helper: build rects for a slice of the page\n * @param geo - page geometry\n * @param from - from index\n * @param to - to index\n * @param merge - whether to merge adjacent rects (default: true)\n * @returns rects\n */\nexport function rectsWithinSlice(\n geo: PdfPageGeometry,\n from: number,\n to: number,\n merge: boolean = true,\n): Rect[] {\n const textRuns: TextRunInfo[] = [];\n\n for (const run of geo.runs) {\n const runStart = run.charStart;\n const runEnd = runStart + run.glyphs.length - 1;\n if (runEnd < from || runStart > to) continue;\n\n const sIdx = Math.max(from, runStart) - runStart;\n const eIdx = Math.min(to, runEnd) - runStart;\n\n let minX = Infinity,\n maxX = -Infinity;\n let minY = Infinity,\n maxY = -Infinity;\n let charCount = 0;\n\n for (let i = sIdx; i <= eIdx; i++) {\n const g = run.glyphs[i];\n if (g.flags === 2) continue; // empty glyph\n\n minX = Math.min(minX, g.x);\n maxX = Math.max(maxX, g.x + g.width);\n minY = Math.min(minY, g.y);\n maxY = Math.max(maxY, g.y + g.height);\n charCount++;\n }\n\n if (minX !== Infinity && charCount > 0) {\n textRuns.push({\n rect: {\n origin: { x: minX, y: minY },\n size: { width: maxX - minX, height: maxY - minY },\n },\n charCount,\n });\n }\n }\n\n // If merge is false, just return the individual rects\n if (!merge) {\n return textRuns.map((run) => run.rect);\n }\n\n // Otherwise merge adjacent rects\n return mergeAdjacentRects(textRuns);\n}\n\n/**\n * ============================================================================\n * Rectangle Merging Algorithm\n * ============================================================================\n *\n * The following code is adapted from Chromium's PDF text selection implementation.\n *\n * Copyright 2010 The Chromium Authors\n * Use of this source code is governed by a BSD-style license that can be\n * found in the LICENSE file: https://source.chromium.org/chromium/chromium/src/+/main:LICENSE\n *\n * Original source:\n * https://source.chromium.org/chromium/chromium/src/+/main:pdf/pdfium/pdfium_range.cc\n *\n * Adapted for TypeScript and this project's Rect/geometry types.\n */\n\n/**\n * Text run info for rect merging (similar to Chromium's ScreenRectTextRunInfo)\n */\nexport interface TextRunInfo {\n rect: Rect;\n charCount: number;\n}\n\n/**\n * Helper functions for Rect operations\n */\nexport function rectUnion(rect1: Rect, rect2: Rect): Rect {\n const left = Math.min(rect1.origin.x, rect2.origin.x);\n const top = Math.min(rect1.origin.y, rect2.origin.y);\n const right = Math.max(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.max(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n return {\n origin: { x: left, y: top },\n size: { width: right - left, height: bottom - top },\n };\n}\n\nexport function rectIntersect(rect1: Rect, rect2: Rect): Rect {\n const left = Math.max(rect1.origin.x, rect2.origin.x);\n const top = Math.max(rect1.origin.y, rect2.origin.y);\n const right = Math.min(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.min(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n const width = Math.max(0, right - left);\n const height = Math.max(0, bottom - top);\n\n return {\n origin: { x: left, y: top },\n size: { width, height },\n };\n}\n\nexport function rectIsEmpty(rect: Rect): boolean {\n return rect.size.width <= 0 || rect.size.height <= 0;\n}\n\n/**\n * Returns a ratio between [0, 1] representing vertical overlap\n */\nexport function getVerticalOverlap(rect1: Rect, rect2: Rect): number {\n if (rectIsEmpty(rect1) || rectIsEmpty(rect2)) return 0;\n\n const unionRect = rectUnion(rect1, rect2);\n\n if (unionRect.size.height === rect1.size.height || unionRect.size.height === rect2.size.height) {\n return 1.0;\n }\n\n const intersectRect = rectIntersect(rect1, rect2);\n return intersectRect.size.height / unionRect.size.height;\n}\n\n/**\n * Returns true if there is sufficient horizontal and vertical overlap\n */\nexport function shouldMergeHorizontalRects(textRun1: TextRunInfo, textRun2: TextRunInfo): boolean {\n const VERTICAL_OVERLAP_THRESHOLD = 0.8;\n const rect1 = textRun1.rect;\n const rect2 = textRun2.rect;\n\n if (getVerticalOverlap(rect1, rect2) < VERTICAL_OVERLAP_THRESHOLD) {\n return false;\n }\n\n const HORIZONTAL_WIDTH_FACTOR = 1.0;\n const averageWidth1 = (HORIZONTAL_WIDTH_FACTOR * rect1.size.width) / textRun1.charCount;\n const averageWidth2 = (HORIZONTAL_WIDTH_FACTOR * rect2.size.width) / textRun2.charCount;\n\n const rect1Left = rect1.origin.x - averageWidth1;\n const rect1Right = rect1.origin.x + rect1.size.width + averageWidth1;\n const rect2Left = rect2.origin.x - averageWidth2;\n const rect2Right = rect2.origin.x + rect2.size.width + averageWidth2;\n\n return rect1Left < rect2Right && rect1Right > rect2Left;\n}\n\n/**\n * Merge adjacent rectangles based on proximity and overlap (similar to Chromium's algorithm)\n */\nexport function mergeAdjacentRects(textRuns: TextRunInfo[]): Rect[] {\n const results: Rect[] = [];\n let previousTextRun: TextRunInfo | null = null;\n let currentRect: Rect | null = null;\n\n for (const textRun of textRuns) {\n if (previousTextRun && currentRect) {\n if (shouldMergeHorizontalRects(previousTextRun, textRun)) {\n currentRect = rectUnion(currentRect, textRun.rect);\n } else {\n results.push(currentRect);\n currentRect = textRun.rect;\n }\n } else {\n currentRect = textRun.rect;\n }\n previousTextRun = textRun;\n }\n\n if (currentRect && !rectIsEmpty(currentRect)) {\n results.push(currentRect);\n }\n\n return results;\n}\n","import {\n BasePlugin,\n PluginRegistry,\n REFRESH_PAGES,\n SET_DOCUMENT,\n Unsubscribe,\n createBehaviorEmitter,\n createEmitter,\n} from '@embedpdf/core';\nimport {\n PdfEngine,\n PdfPageGeometry,\n Rect,\n PdfTask,\n PdfTaskHelper,\n PdfErrorCode,\n ignore,\n PageTextSlice,\n Task,\n} from '@embedpdf/models';\n\nimport {\n cachePageGeometry,\n setSelection,\n SelectionAction,\n endSelection,\n startSelection,\n clearSelection,\n reset,\n setRects,\n setSlices,\n} from './actions';\nimport * as selector from './selectors';\nimport {\n SelectionCapability,\n SelectionPluginConfig,\n SelectionRangeX,\n SelectionState,\n} from './types';\nimport { sliceBounds, rectsWithinSlice } from './utils';\n\nexport class SelectionPlugin extends BasePlugin<\n SelectionPluginConfig,\n SelectionCapability,\n SelectionState,\n SelectionAction\n> {\n static readonly id = 'selection' as const;\n\n /** Modes that should trigger text-selection logic */\n private enabledModes = new Set<string>(['pointerMode']);\n\n /* interactive state */\n private selecting = false;\n private anchor?: { page: number; index: number };\n\n private readonly selChange$ = createBehaviorEmitter<SelectionState['selection']>();\n private readonly textRetrieved$ = createBehaviorEmitter<string[]>();\n private readonly copyToClipboard$ = createEmitter<string>();\n private readonly beginSelection$ = createEmitter<{ page: number; index: number }>();\n private readonly endSelection$ = createEmitter<void>();\n private readonly refreshPages$ = createEmitter<number[]>();\n\n constructor(id: string, registry: PluginRegistry) {\n super(id, registry);\n\n this.coreStore.onAction(SET_DOCUMENT, (_action) => {\n this.dispatch(reset());\n });\n\n this.coreStore.onAction(REFRESH_PAGES, (action) => {\n const tasks = action.payload.map((pageIdx) => this.getNewPageGeometryAndCache(pageIdx));\n Task.all(tasks).wait(() => {\n this.refreshPages$.emit(action.payload);\n }, ignore);\n });\n }\n\n /* ── life-cycle ────────────────────────────────────────── */\n async initialize() {}\n async destroy() {\n this.selChange$.clear();\n }\n\n /* ── capability exposed to UI / other plugins ─────────── */\n buildCapability(): SelectionCapability {\n return {\n getGeometry: (p) => this.getOrLoadGeometry(p),\n getFormattedSelection: () => selector.getFormattedSelection(this.state),\n getFormattedSelectionForPage: (p) => selector.getFormattedSelectionForPage(this.state, p),\n getHighlightRectsForPage: (p) => selector.selectRectsForPage(this.state, p),\n getHighlightRects: () => this.state.rects,\n getBoundingRectForPage: (p) => selector.selectBoundingRectForPage(this.state, p),\n getBoundingRects: () => selector.selectBoundingRectsForAllPages(this.state),\n begin: (p, i) => this.beginSelection(p, i),\n update: (p, i) => this.updateSelection(p, i),\n end: () => this.endSelection(),\n clear: () => this.clearSelection(),\n onCopyToClipboard: this.copyToClipboard$.on,\n onSelectionChange: this.selChange$.on,\n onTextRetrieved: this.textRetrieved$.on,\n onBeginSelection: this.beginSelection$.on,\n onEndSelection: this.endSelection$.on,\n getSelectedText: () => this.getSelectedText(),\n copyToClipboard: () => this.copyToClipboard(),\n enableForMode: (id: string) => this.enabledModes.add(id),\n isEnabledForMode: (id: string) => this.enabledModes.has(id),\n getState: () => this.state,\n };\n }\n\n public onRefreshPages(fn: (pages: number[]) => void): Unsubscribe {\n return this.refreshPages$.on(fn);\n }\n\n private getNewPageGeometryAndCache(pageIdx: number): PdfTask<PdfPageGeometry> {\n if (!this.coreState.core.document)\n return PdfTaskHelper.reject({ code: PdfErrorCode.NotFound, message: 'Doc Not Found' });\n const page = this.coreState.core.document.pages.find((p) => p.index === pageIdx)!;\n const task = this.engine.getPageGeometry(this.coreState.core.document, page);\n task.wait((geo) => {\n this.dispatch(cachePageGeometry(pageIdx, geo));\n }, ignore);\n return task;\n }\n\n /* ── geometry cache ───────────────────────────────────── */\n private getOrLoadGeometry(pageIdx: number): PdfTask<PdfPageGeometry> {\n const cached = this.state.geometry[pageIdx];\n if (cached) return PdfTaskHelper.resolve(cached);\n\n return this.getNewPageGeometryAndCache(pageIdx);\n }\n\n /* ── selection state updates ───────────────────────────── */\n private beginSelection(page: number, index: number) {\n this.selecting = true;\n this.anchor = { page, index };\n this.dispatch(startSelection());\n this.beginSelection$.emit({ page, index });\n }\n\n private endSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(endSelection());\n this.endSelection$.emit();\n }\n\n private clearSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(clearSelection());\n this.selChange$.emit(null);\n }\n\n private updateSelection(page: number, index: number) {\n if (!this.selecting || !this.anchor) return;\n\n const a = this.anchor;\n const forward = page > a.page || (page === a.page && index >= a.index);\n\n const start = forward ? a : { page, index };\n const end = forward ? { page, index } : a;\n\n const range = { start, end };\n this.dispatch(setSelection(range));\n this.updateRectsAndSlices(range);\n this.selChange$.emit(range);\n }\n\n private updateRectsAndSlices(range: SelectionRangeX) {\n const allRects: Record<number, Rect[]> = {};\n const allSlices: Record<number, { start: number; count: number }> = {};\n\n for (let p = range.start.page; p <= range.end.page; p++) {\n const geo = this.state.geometry[p];\n const sb = sliceBounds(range, geo, p);\n if (!sb) continue;\n\n allRects[p] = rectsWithinSlice(geo!, sb.from, sb.to);\n allSlices[p] = { start: sb.from, count: sb.to - sb.from + 1 };\n }\n\n this.dispatch(setRects(allRects));\n this.dispatch(setSlices(allSlices));\n }\n\n private getSelectedText(): PdfTask<string[]> {\n if (!this.coreState.core.document || !this.state.selection) {\n return PdfTaskHelper.reject({\n code: PdfErrorCode.NotFound,\n message: 'Doc Not Found or No Selection',\n });\n }\n\n const sel = this.state.selection;\n const req: PageTextSlice[] = [];\n\n for (let p = sel.start.page; p <= sel.end.page; p++) {\n const s = this.state.slices[p];\n if (s) req.push({ pageIndex: p, charIndex: s.start, charCount: s.count });\n }\n\n if (req.length === 0) return PdfTaskHelper.resolve([] as string[]);\n\n const task = this.engine.getTextSlices(this.coreState.core.document, req);\n\n // Emit the text when it's retrieved\n task.wait((text) => {\n this.textRetrieved$.emit(text);\n }, ignore);\n\n return task;\n }\n\n private copyToClipboard() {\n const text = this.getSelectedText();\n text.wait((text) => {\n this.copyToClipboard$.emit(text.join('\\n'));\n }, ignore);\n }\n}\n","import { SelectionState } from './types';\nimport {\n SelectionAction,\n CACHE_PAGE_GEOMETRY,\n SET_SELECTION,\n START_SELECTION,\n END_SELECTION,\n CLEAR_SELECTION,\n RESET,\n SET_SLICES,\n SET_RECTS,\n} from './actions';\n\nexport const initialState: SelectionState = {\n geometry: {},\n rects: {},\n slices: {},\n selection: null,\n active: false,\n selecting: false,\n};\n\nexport const selectionReducer = (state = initialState, action: SelectionAction): SelectionState => {\n switch (action.type) {\n case CACHE_PAGE_GEOMETRY: {\n const { page, geo } = action.payload;\n return { ...state, geometry: { ...state.geometry, [page]: geo } };\n }\n case SET_SELECTION:\n return { ...state, selection: action.payload, active: true };\n case START_SELECTION:\n return { ...state, selecting: true, selection: null, rects: {} };\n case END_SELECTION:\n return { ...state, selecting: false };\n case CLEAR_SELECTION:\n return { ...state, selecting: false, selection: null, rects: {}, active: false };\n case SET_RECTS:\n return { ...state, rects: action.payload };\n case SET_SLICES:\n return { ...state, slices: action.payload };\n case RESET:\n return initialState;\n default:\n return state;\n }\n};\n","import { PluginPackage } from '@embedpdf/core';\nimport { manifest, SELECTION_PLUGIN_ID } from './manifest';\nimport { SelectionPluginConfig, SelectionState } from './types';\n\nimport { SelectionPlugin } from './selection-plugin';\nimport { SelectionAction } from './actions';\nimport { selectionReducer, initialState } from './reducer';\n\nexport const SelectionPluginPackage: PluginPackage<\n SelectionPlugin,\n SelectionPluginConfig,\n SelectionState,\n SelectionAction\n> = {\n manifest,\n create: (registry) => new SelectionPlugin(SELECTION_PLUGIN_ID, registry),\n reducer: selectionReducer,\n initialState,\n};\n\nexport * from './selection-plugin';\nexport * from './types';\nexport * from './manifest';\nexport * from './utils';\n"],"names":["SELECTION_PLUGIN_ID","manifest","id","name","version","provides","requires","optional","defaultConfig","enabled","CACHE_PAGE_GEOMETRY","SET_SELECTION","START_SELECTION","END_SELECTION","CLEAR_SELECTION","SET_RECTS","SET_SLICES","RESET","selectRectsForPage","state","page","rects","selectBoundingRectForPage","boundingRect","sliceBounds","sel","geo","start","end","from","index","lastRun","runs","length","lastCharOnPage","charStart","glyphs","to","rectsWithinSlice","merge","textRuns","run","runStart","runEnd","sIdx","Math","max","eIdx","min","minX","Infinity","maxX","minY","maxY","charCount","i","g","flags","x","width","y","height","push","rect","origin","size","mergeAdjacentRects","map","rectUnion","rect1","rect2","left","top","rectIntersect","right","bottom","rectIsEmpty","getVerticalOverlap","unionRect","shouldMergeHorizontalRects","textRun1","textRun2","averageWidth1","averageWidth2","rect1Left","rect1Right","rect2Left","results","previousTextRun","currentRect","textRun","_SelectionPlugin","BasePlugin","constructor","registry","super","this","enabledModes","Set","selecting","selChange$","createBehaviorEmitter","textRetrieved$","copyToClipboard$","createEmitter","beginSelection$","endSelection$","refreshPages$","coreStore","onAction","SET_DOCUMENT","_action","dispatch","type","REFRESH_PAGES","action","tasks","payload","pageIdx","getNewPageGeometryAndCache","Task","all","wait","emit","ignore","initialize","destroy","clear","buildCapability","getGeometry","p","getOrLoadGeometry","getFormattedSelection","result","pages","Object","keys","Number","pageIndex","segmentRects","selector.getFormattedSelection","getFormattedSelectionForPage","selector.getFormattedSelectionForPage","getHighlightRectsForPage","selector.selectRectsForPage","getHighlightRects","getBoundingRectForPage","selector.selectBoundingRectForPage","getBoundingRects","out","rectMap","key","bRect","selector.selectBoundingRectsForAllPages","begin","beginSelection","update","updateSelection","endSelection","clearSelection","onCopyToClipboard","on","onSelectionChange","onTextRetrieved","onBeginSelection","onEndSelection","getSelectedText","copyToClipboard","enableForMode","add","isEnabledForMode","has","getState","onRefreshPages","fn","coreState","core","document","PdfTaskHelper","reject","code","PdfErrorCode","NotFound","message","find","task","engine","getPageGeometry","cachePageGeometry","cached","geometry","resolve","anchor","a","forward","range","updateRectsAndSlices","allRects","allSlices","sb","count","setRects","selection","req","s","slices","charIndex","getTextSlices","text","join","SelectionPlugin","initialState","active","SelectionPluginPackage","create","reducer","pt","rel","findIndex"],"mappings":"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"}
package/dist/index.js CHANGED
@@ -1,5 +1,5 @@
1
- import { BasePlugin, createBehaviorEmitter, createEmitter, SET_DOCUMENT } from "@embedpdf/core";
2
- import { boundingRect, PdfTaskHelper, PdfErrorCode, ignore } from "@embedpdf/models";
1
+ import { BasePlugin, createBehaviorEmitter, createEmitter, SET_DOCUMENT, REFRESH_PAGES } from "@embedpdf/core";
2
+ import { boundingRect, Task, ignore, PdfTaskHelper, PdfErrorCode } from "@embedpdf/models";
3
3
  const SELECTION_PLUGIN_ID = "selection";
4
4
  const manifest = {
5
5
  id: SELECTION_PLUGIN_ID,
@@ -207,9 +207,8 @@ function mergeAdjacentRects(textRuns) {
207
207
  return results;
208
208
  }
209
209
  const _SelectionPlugin = class _SelectionPlugin extends BasePlugin {
210
- constructor(id, registry, engine) {
210
+ constructor(id, registry) {
211
211
  super(id, registry);
212
- this.engine = engine;
213
212
  this.enabledModes = /* @__PURE__ */ new Set(["pointerMode"]);
214
213
  this.selecting = false;
215
214
  this.selChange$ = createBehaviorEmitter();
@@ -217,9 +216,16 @@ const _SelectionPlugin = class _SelectionPlugin extends BasePlugin {
217
216
  this.copyToClipboard$ = createEmitter();
218
217
  this.beginSelection$ = createEmitter();
219
218
  this.endSelection$ = createEmitter();
219
+ this.refreshPages$ = createEmitter();
220
220
  this.coreStore.onAction(SET_DOCUMENT, (_action) => {
221
221
  this.dispatch(reset());
222
222
  });
223
+ this.coreStore.onAction(REFRESH_PAGES, (action) => {
224
+ const tasks = action.payload.map((pageIdx) => this.getNewPageGeometryAndCache(pageIdx));
225
+ Task.all(tasks).wait(() => {
226
+ this.refreshPages$.emit(action.payload);
227
+ }, ignore);
228
+ });
223
229
  }
224
230
  /* ── life-cycle ────────────────────────────────────────── */
225
231
  async initialize() {
@@ -253,10 +259,10 @@ const _SelectionPlugin = class _SelectionPlugin extends BasePlugin {
253
259
  getState: () => this.state
254
260
  };
255
261
  }
256
- /* ── geometry cache ───────────────────────────────────── */
257
- getOrLoadGeometry(pageIdx) {
258
- const cached = this.state.geometry[pageIdx];
259
- if (cached) return PdfTaskHelper.resolve(cached);
262
+ onRefreshPages(fn) {
263
+ return this.refreshPages$.on(fn);
264
+ }
265
+ getNewPageGeometryAndCache(pageIdx) {
260
266
  if (!this.coreState.core.document)
261
267
  return PdfTaskHelper.reject({ code: PdfErrorCode.NotFound, message: "Doc Not Found" });
262
268
  const page = this.coreState.core.document.pages.find((p) => p.index === pageIdx);
@@ -266,6 +272,12 @@ const _SelectionPlugin = class _SelectionPlugin extends BasePlugin {
266
272
  }, ignore);
267
273
  return task;
268
274
  }
275
+ /* ── geometry cache ───────────────────────────────────── */
276
+ getOrLoadGeometry(pageIdx) {
277
+ const cached = this.state.geometry[pageIdx];
278
+ if (cached) return PdfTaskHelper.resolve(cached);
279
+ return this.getNewPageGeometryAndCache(pageIdx);
280
+ }
269
281
  /* ── selection state updates ───────────────────────────── */
270
282
  beginSelection(page, index) {
271
283
  this.selecting = true;
@@ -372,7 +384,7 @@ const selectionReducer = (state = initialState, action) => {
372
384
  };
373
385
  const SelectionPluginPackage = {
374
386
  manifest,
375
- create: (registry, engine) => new SelectionPlugin(SELECTION_PLUGIN_ID, registry, engine),
387
+ create: (registry) => new SelectionPlugin(SELECTION_PLUGIN_ID, registry),
376
388
  reducer: selectionReducer,
377
389
  initialState
378
390
  };
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/lib/manifest.ts","../src/lib/actions.ts","../src/lib/selectors.ts","../src/lib/utils.ts","../src/lib/selection-plugin.ts","../src/lib/reducer.ts","../src/lib/index.ts"],"sourcesContent":["import { PluginManifest } from '@embedpdf/core';\nimport { SelectionPluginConfig } from './types';\n\nexport const SELECTION_PLUGIN_ID = 'selection';\n\nexport const manifest: PluginManifest<SelectionPluginConfig> = {\n id: SELECTION_PLUGIN_ID,\n name: 'Selection Plugin',\n version: '1.0.0',\n provides: ['selection'],\n requires: ['interaction-manager'],\n optional: [],\n defaultConfig: {\n enabled: true,\n },\n};\n","import { Action } from '@embedpdf/core';\nimport { PdfPageGeometry, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\nexport const CACHE_PAGE_GEOMETRY = 'CACHE_PAGE_GEOMETRY';\nexport const SET_SELECTION = 'SET_SELECTION';\nexport const START_SELECTION = 'START_SELECTION';\nexport const END_SELECTION = 'END_SELECTION';\nexport const CLEAR_SELECTION = 'CLEAR_SELECTION';\nexport const SET_RECTS = 'SET_RECTS';\nexport const SET_SLICES = 'SET_SLICES';\nexport const RESET = 'RESET';\n\nexport interface CachePageGeometryAction extends Action {\n type: typeof CACHE_PAGE_GEOMETRY;\n payload: { page: number; geo: PdfPageGeometry };\n}\nexport interface SetSelectionAction extends Action {\n type: typeof SET_SELECTION;\n payload: SelectionRangeX | null;\n}\n\nexport interface StartSelectionAction extends Action {\n type: typeof START_SELECTION;\n}\n\nexport interface EndSelectionAction extends Action {\n type: typeof END_SELECTION;\n}\n\nexport interface ClearSelectionAction extends Action {\n type: typeof CLEAR_SELECTION;\n}\n\nexport interface SetRectsAction extends Action {\n type: typeof SET_RECTS;\n payload: Record<number, Rect[]>;\n}\n\nexport interface SetSlicesAction extends Action {\n type: typeof SET_SLICES;\n payload: Record<number, { start: number; count: number }>;\n}\n\nexport interface ResetAction extends Action {\n type: typeof RESET;\n}\n\nexport type SelectionAction =\n | CachePageGeometryAction\n | SetSelectionAction\n | StartSelectionAction\n | EndSelectionAction\n | ClearSelectionAction\n | SetRectsAction\n | SetSlicesAction\n | ResetAction;\n\nexport const cachePageGeometry = (page: number, geo: PdfPageGeometry): CachePageGeometryAction => ({\n type: CACHE_PAGE_GEOMETRY,\n payload: { page, geo },\n});\n\nexport const setSelection = (sel: SelectionRangeX): SetSelectionAction => ({\n type: SET_SELECTION,\n payload: sel,\n});\n\nexport const startSelection = (): StartSelectionAction => ({ type: START_SELECTION });\n\nexport const endSelection = (): EndSelectionAction => ({ type: END_SELECTION });\n\nexport const clearSelection = (): ClearSelectionAction => ({ type: CLEAR_SELECTION });\n\nexport const setRects = (allRects: Record<number, Rect[]>): SetRectsAction => ({\n type: SET_RECTS,\n payload: allRects,\n});\n\nexport const setSlices = (\n slices: Record<number, { start: number; count: number }>,\n): SetSlicesAction => ({ type: SET_SLICES, payload: slices });\n\nexport const reset = (): ResetAction => ({ type: RESET });\n","import { Rect, boundingRect } from '@embedpdf/models';\nimport { FormattedSelection, SelectionState } from './types';\n\nexport function selectRectsForPage(state: SelectionState, page: number) {\n return state.rects[page] ?? [];\n}\n\nexport function selectBoundingRectForPage(state: SelectionState, page: number) {\n return boundingRect(selectRectsForPage(state, page));\n}\n\nexport function selectRectsAndBoundingRectForPage(state: SelectionState, page: number) {\n return {\n rects: selectRectsForPage(state, page),\n boundingRect: selectBoundingRectForPage(state, page),\n };\n}\n\nexport function selectBoundingRectsForAllPages(state: SelectionState) {\n const out: { page: number; rect: Rect }[] = [];\n const rectMap = state.rects;\n\n for (const key in rectMap) {\n const page = Number(key);\n const bRect = boundingRect(rectMap[page]);\n if (bRect) out.push({ page, rect: bRect });\n }\n return out;\n}\n\nexport function getFormattedSelectionForPage(\n state: SelectionState,\n page: number,\n): FormattedSelection | null {\n const segmentRects = state.rects[page] || [];\n if (segmentRects.length === 0) return null;\n const boundingRect = selectBoundingRectForPage(state, page);\n if (!boundingRect) return null;\n return { pageIndex: page, rect: boundingRect, segmentRects };\n}\n\nexport function getFormattedSelection(state: SelectionState) {\n const result: FormattedSelection[] = [];\n\n // Get all pages that have rects\n const pages = Object.keys(state.rects).map(Number);\n\n for (const pageIndex of pages) {\n const segmentRects = state.rects[pageIndex] || [];\n\n if (segmentRects.length === 0) continue;\n\n // Calculate bounding rect for this page\n const boundingRect = selectBoundingRectForPage(state, pageIndex);\n\n if (boundingRect) {\n result.push({\n pageIndex,\n rect: boundingRect,\n segmentRects,\n });\n }\n }\n\n return result;\n}\n","import { PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\n/**\n * Hit-test helper using runs\n * @param geo - page geometry\n * @param pt - point\n * @returns glyph index\n */\nexport function glyphAt(geo: PdfPageGeometry, pt: Position) {\n for (const run of geo.runs) {\n const inRun =\n pt.y >= run.rect.y &&\n pt.y <= run.rect.y + run.rect.height &&\n pt.x >= run.rect.x &&\n pt.x <= run.rect.x + run.rect.width;\n\n if (!inRun) continue;\n\n // Simply check if the point is within any glyph's bounding box\n const rel = run.glyphs.findIndex(\n (g) => pt.x >= g.x && pt.x <= g.x + g.width && pt.y >= g.y && pt.y <= g.y + g.height,\n );\n\n if (rel !== -1) {\n return run.charStart + rel;\n }\n }\n return -1;\n}\n\n/**\n * Helper: min/max glyph indices on `page` for current sel\n * @param sel - selection range\n * @param geo - page geometry\n * @param page - page index\n * @returns { from: number; to: number } | null\n */\nexport function sliceBounds(\n sel: SelectionRangeX | null,\n geo: PdfPageGeometry | undefined,\n page: number,\n): { from: number; to: number } | null {\n if (!sel || !geo) return null;\n if (page < sel.start.page || page > sel.end.page) return null;\n\n const from = page === sel.start.page ? sel.start.index : 0;\n\n const lastRun = geo.runs[geo.runs.length - 1];\n const lastCharOnPage = lastRun.charStart + lastRun.glyphs.length - 1;\n\n const to = page === sel.end.page ? sel.end.index : lastCharOnPage;\n\n return { from, to };\n}\n\n/**\n * Helper: build rects for a slice of the page\n * @param geo - page geometry\n * @param from - from index\n * @param to - to index\n * @param merge - whether to merge adjacent rects (default: true)\n * @returns rects\n */\nexport function rectsWithinSlice(\n geo: PdfPageGeometry,\n from: number,\n to: number,\n merge: boolean = true,\n): Rect[] {\n const textRuns: TextRunInfo[] = [];\n\n for (const run of geo.runs) {\n const runStart = run.charStart;\n const runEnd = runStart + run.glyphs.length - 1;\n if (runEnd < from || runStart > to) continue;\n\n const sIdx = Math.max(from, runStart) - runStart;\n const eIdx = Math.min(to, runEnd) - runStart;\n\n let minX = Infinity,\n maxX = -Infinity;\n let minY = Infinity,\n maxY = -Infinity;\n let charCount = 0;\n\n for (let i = sIdx; i <= eIdx; i++) {\n const g = run.glyphs[i];\n if (g.flags === 2) continue; // empty glyph\n\n minX = Math.min(minX, g.x);\n maxX = Math.max(maxX, g.x + g.width);\n minY = Math.min(minY, g.y);\n maxY = Math.max(maxY, g.y + g.height);\n charCount++;\n }\n\n if (minX !== Infinity && charCount > 0) {\n textRuns.push({\n rect: {\n origin: { x: minX, y: minY },\n size: { width: maxX - minX, height: maxY - minY },\n },\n charCount,\n });\n }\n }\n\n // If merge is false, just return the individual rects\n if (!merge) {\n return textRuns.map((run) => run.rect);\n }\n\n // Otherwise merge adjacent rects\n return mergeAdjacentRects(textRuns);\n}\n\n/**\n * ============================================================================\n * Rectangle Merging Algorithm\n * ============================================================================\n *\n * The following code is adapted from Chromium's PDF text selection implementation.\n *\n * Copyright 2010 The Chromium Authors\n * Use of this source code is governed by a BSD-style license that can be\n * found in the LICENSE file: https://source.chromium.org/chromium/chromium/src/+/main:LICENSE\n *\n * Original source:\n * https://source.chromium.org/chromium/chromium/src/+/main:pdf/pdfium/pdfium_range.cc\n *\n * Adapted for TypeScript and this project's Rect/geometry types.\n */\n\n/**\n * Text run info for rect merging (similar to Chromium's ScreenRectTextRunInfo)\n */\nexport interface TextRunInfo {\n rect: Rect;\n charCount: number;\n}\n\n/**\n * Helper functions for Rect operations\n */\nexport function rectUnion(rect1: Rect, rect2: Rect): Rect {\n const left = Math.min(rect1.origin.x, rect2.origin.x);\n const top = Math.min(rect1.origin.y, rect2.origin.y);\n const right = Math.max(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.max(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n return {\n origin: { x: left, y: top },\n size: { width: right - left, height: bottom - top },\n };\n}\n\nexport function rectIntersect(rect1: Rect, rect2: Rect): Rect {\n const left = Math.max(rect1.origin.x, rect2.origin.x);\n const top = Math.max(rect1.origin.y, rect2.origin.y);\n const right = Math.min(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.min(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n const width = Math.max(0, right - left);\n const height = Math.max(0, bottom - top);\n\n return {\n origin: { x: left, y: top },\n size: { width, height },\n };\n}\n\nexport function rectIsEmpty(rect: Rect): boolean {\n return rect.size.width <= 0 || rect.size.height <= 0;\n}\n\n/**\n * Returns a ratio between [0, 1] representing vertical overlap\n */\nexport function getVerticalOverlap(rect1: Rect, rect2: Rect): number {\n if (rectIsEmpty(rect1) || rectIsEmpty(rect2)) return 0;\n\n const unionRect = rectUnion(rect1, rect2);\n\n if (unionRect.size.height === rect1.size.height || unionRect.size.height === rect2.size.height) {\n return 1.0;\n }\n\n const intersectRect = rectIntersect(rect1, rect2);\n return intersectRect.size.height / unionRect.size.height;\n}\n\n/**\n * Returns true if there is sufficient horizontal and vertical overlap\n */\nexport function shouldMergeHorizontalRects(textRun1: TextRunInfo, textRun2: TextRunInfo): boolean {\n const VERTICAL_OVERLAP_THRESHOLD = 0.8;\n const rect1 = textRun1.rect;\n const rect2 = textRun2.rect;\n\n if (getVerticalOverlap(rect1, rect2) < VERTICAL_OVERLAP_THRESHOLD) {\n return false;\n }\n\n const HORIZONTAL_WIDTH_FACTOR = 1.0;\n const averageWidth1 = (HORIZONTAL_WIDTH_FACTOR * rect1.size.width) / textRun1.charCount;\n const averageWidth2 = (HORIZONTAL_WIDTH_FACTOR * rect2.size.width) / textRun2.charCount;\n\n const rect1Left = rect1.origin.x - averageWidth1;\n const rect1Right = rect1.origin.x + rect1.size.width + averageWidth1;\n const rect2Left = rect2.origin.x - averageWidth2;\n const rect2Right = rect2.origin.x + rect2.size.width + averageWidth2;\n\n return rect1Left < rect2Right && rect1Right > rect2Left;\n}\n\n/**\n * Merge adjacent rectangles based on proximity and overlap (similar to Chromium's algorithm)\n */\nexport function mergeAdjacentRects(textRuns: TextRunInfo[]): Rect[] {\n const results: Rect[] = [];\n let previousTextRun: TextRunInfo | null = null;\n let currentRect: Rect | null = null;\n\n for (const textRun of textRuns) {\n if (previousTextRun && currentRect) {\n if (shouldMergeHorizontalRects(previousTextRun, textRun)) {\n currentRect = rectUnion(currentRect, textRun.rect);\n } else {\n results.push(currentRect);\n currentRect = textRun.rect;\n }\n } else {\n currentRect = textRun.rect;\n }\n previousTextRun = textRun;\n }\n\n if (currentRect && !rectIsEmpty(currentRect)) {\n results.push(currentRect);\n }\n\n return results;\n}\n","import {\n BasePlugin,\n PluginRegistry,\n SET_DOCUMENT,\n createBehaviorEmitter,\n createEmitter,\n} from '@embedpdf/core';\nimport {\n PdfEngine,\n PdfPageGeometry,\n Rect,\n PdfTask,\n PdfTaskHelper,\n PdfErrorCode,\n ignore,\n PageTextSlice,\n} from '@embedpdf/models';\n\nimport {\n cachePageGeometry,\n setSelection,\n SelectionAction,\n endSelection,\n startSelection,\n clearSelection,\n reset,\n setRects,\n setSlices,\n} from './actions';\nimport * as selector from './selectors';\nimport {\n SelectionCapability,\n SelectionPluginConfig,\n SelectionRangeX,\n SelectionState,\n} from './types';\nimport { sliceBounds, rectsWithinSlice } from './utils';\n\nexport class SelectionPlugin extends BasePlugin<\n SelectionPluginConfig,\n SelectionCapability,\n SelectionState,\n SelectionAction\n> {\n static readonly id = 'selection' as const;\n\n /** Modes that should trigger text-selection logic */\n private enabledModes = new Set<string>(['pointerMode']);\n\n /* interactive state */\n private selecting = false;\n private anchor?: { page: number; index: number };\n\n private readonly selChange$ = createBehaviorEmitter<SelectionState['selection']>();\n private readonly textRetrieved$ = createBehaviorEmitter<string[]>();\n private readonly copyToClipboard$ = createEmitter<string>();\n private readonly beginSelection$ = createEmitter<{ page: number; index: number }>();\n private readonly endSelection$ = createEmitter<void>();\n\n constructor(\n id: string,\n registry: PluginRegistry,\n private engine: PdfEngine,\n ) {\n super(id, registry);\n\n this.coreStore.onAction(SET_DOCUMENT, (_action) => {\n this.dispatch(reset());\n });\n }\n\n /* ── life-cycle ────────────────────────────────────────── */\n async initialize() {}\n async destroy() {\n this.selChange$.clear();\n }\n\n /* ── capability exposed to UI / other plugins ─────────── */\n buildCapability(): SelectionCapability {\n return {\n getGeometry: (p) => this.getOrLoadGeometry(p),\n getFormattedSelection: () => selector.getFormattedSelection(this.state),\n getFormattedSelectionForPage: (p) => selector.getFormattedSelectionForPage(this.state, p),\n getHighlightRectsForPage: (p) => selector.selectRectsForPage(this.state, p),\n getHighlightRects: () => this.state.rects,\n getBoundingRectForPage: (p) => selector.selectBoundingRectForPage(this.state, p),\n getBoundingRects: () => selector.selectBoundingRectsForAllPages(this.state),\n begin: (p, i) => this.beginSelection(p, i),\n update: (p, i) => this.updateSelection(p, i),\n end: () => this.endSelection(),\n clear: () => this.clearSelection(),\n onCopyToClipboard: this.copyToClipboard$.on,\n onSelectionChange: this.selChange$.on,\n onTextRetrieved: this.textRetrieved$.on,\n onBeginSelection: this.beginSelection$.on,\n onEndSelection: this.endSelection$.on,\n getSelectedText: () => this.getSelectedText(),\n copyToClipboard: () => this.copyToClipboard(),\n enableForMode: (id: string) => this.enabledModes.add(id),\n isEnabledForMode: (id: string) => this.enabledModes.has(id),\n getState: () => this.state,\n };\n }\n\n /* ── geometry cache ───────────────────────────────────── */\n private getOrLoadGeometry(pageIdx: number): PdfTask<PdfPageGeometry> {\n const cached = this.state.geometry[pageIdx];\n if (cached) return PdfTaskHelper.resolve(cached);\n\n if (!this.coreState.core.document)\n return PdfTaskHelper.reject({ code: PdfErrorCode.NotFound, message: 'Doc Not Found' });\n const page = this.coreState.core.document.pages.find((p) => p.index === pageIdx)!;\n\n const task = this.engine.getPageGeometry(this.coreState.core.document, page);\n\n task.wait((geo) => {\n this.dispatch(cachePageGeometry(pageIdx, geo));\n }, ignore);\n\n return task;\n }\n\n /* ── selection state updates ───────────────────────────── */\n private beginSelection(page: number, index: number) {\n this.selecting = true;\n this.anchor = { page, index };\n this.dispatch(startSelection());\n this.beginSelection$.emit({ page, index });\n }\n\n private endSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(endSelection());\n this.endSelection$.emit();\n }\n\n private clearSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(clearSelection());\n this.selChange$.emit(null);\n }\n\n private updateSelection(page: number, index: number) {\n if (!this.selecting || !this.anchor) return;\n\n const a = this.anchor;\n const forward = page > a.page || (page === a.page && index >= a.index);\n\n const start = forward ? a : { page, index };\n const end = forward ? { page, index } : a;\n\n const range = { start, end };\n this.dispatch(setSelection(range));\n this.updateRectsAndSlices(range);\n this.selChange$.emit(range);\n }\n\n private updateRectsAndSlices(range: SelectionRangeX) {\n const allRects: Record<number, Rect[]> = {};\n const allSlices: Record<number, { start: number; count: number }> = {};\n\n for (let p = range.start.page; p <= range.end.page; p++) {\n const geo = this.state.geometry[p];\n const sb = sliceBounds(range, geo, p);\n if (!sb) continue;\n\n allRects[p] = rectsWithinSlice(geo!, sb.from, sb.to);\n allSlices[p] = { start: sb.from, count: sb.to - sb.from + 1 };\n }\n\n this.dispatch(setRects(allRects));\n this.dispatch(setSlices(allSlices));\n }\n\n private getSelectedText(): PdfTask<string[]> {\n if (!this.coreState.core.document || !this.state.selection) {\n return PdfTaskHelper.reject({\n code: PdfErrorCode.NotFound,\n message: 'Doc Not Found or No Selection',\n });\n }\n\n const sel = this.state.selection;\n const req: PageTextSlice[] = [];\n\n for (let p = sel.start.page; p <= sel.end.page; p++) {\n const s = this.state.slices[p];\n if (s) req.push({ pageIndex: p, charIndex: s.start, charCount: s.count });\n }\n\n if (req.length === 0) return PdfTaskHelper.resolve([] as string[]);\n\n const task = this.engine.getTextSlices(this.coreState.core.document, req);\n\n // Emit the text when it's retrieved\n task.wait((text) => {\n this.textRetrieved$.emit(text);\n }, ignore);\n\n return task;\n }\n\n private copyToClipboard() {\n const text = this.getSelectedText();\n text.wait((text) => {\n this.copyToClipboard$.emit(text.join('\\n'));\n }, ignore);\n }\n}\n","import { SelectionState } from './types';\nimport {\n SelectionAction,\n CACHE_PAGE_GEOMETRY,\n SET_SELECTION,\n START_SELECTION,\n END_SELECTION,\n CLEAR_SELECTION,\n RESET,\n SET_SLICES,\n SET_RECTS,\n} from './actions';\n\nexport const initialState: SelectionState = {\n geometry: {},\n rects: {},\n slices: {},\n selection: null,\n active: false,\n selecting: false,\n};\n\nexport const selectionReducer = (state = initialState, action: SelectionAction): SelectionState => {\n switch (action.type) {\n case CACHE_PAGE_GEOMETRY: {\n const { page, geo } = action.payload;\n return { ...state, geometry: { ...state.geometry, [page]: geo } };\n }\n case SET_SELECTION:\n return { ...state, selection: action.payload, active: true };\n case START_SELECTION:\n return { ...state, selecting: true, selection: null, rects: {} };\n case END_SELECTION:\n return { ...state, selecting: false };\n case CLEAR_SELECTION:\n return { ...state, selecting: false, selection: null, rects: {}, active: false };\n case SET_RECTS:\n return { ...state, rects: action.payload };\n case SET_SLICES:\n return { ...state, slices: action.payload };\n case RESET:\n return initialState;\n default:\n return state;\n }\n};\n","import { PluginPackage } from '@embedpdf/core';\nimport { manifest, SELECTION_PLUGIN_ID } from './manifest';\nimport { SelectionPluginConfig, SelectionState } from './types';\n\nimport { SelectionPlugin } from './selection-plugin';\nimport { SelectionAction } from './actions';\nimport { selectionReducer, initialState } from './reducer';\n\nexport const SelectionPluginPackage: PluginPackage<\n SelectionPlugin,\n SelectionPluginConfig,\n SelectionState,\n SelectionAction\n> = {\n manifest,\n create: (registry, engine) => new SelectionPlugin(SELECTION_PLUGIN_ID, registry, engine),\n reducer: selectionReducer,\n initialState,\n};\n\nexport * from './selection-plugin';\nexport * from './types';\nexport * from './manifest';\nexport * from './utils';\n"],"names":["boundingRect","selector.getFormattedSelection","selector.getFormattedSelectionForPage","selector.selectRectsForPage","selector.selectBoundingRectForPage","selector.selectBoundingRectsForAllPages","text"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../src/lib/manifest.ts","../src/lib/actions.ts","../src/lib/selectors.ts","../src/lib/utils.ts","../src/lib/selection-plugin.ts","../src/lib/reducer.ts","../src/lib/index.ts"],"sourcesContent":["import { PluginManifest } from '@embedpdf/core';\nimport { SelectionPluginConfig } from './types';\n\nexport const SELECTION_PLUGIN_ID = 'selection';\n\nexport const manifest: PluginManifest<SelectionPluginConfig> = {\n id: SELECTION_PLUGIN_ID,\n name: 'Selection Plugin',\n version: '1.0.0',\n provides: ['selection'],\n requires: ['interaction-manager'],\n optional: [],\n defaultConfig: {\n enabled: true,\n },\n};\n","import { Action } from '@embedpdf/core';\nimport { PdfPageGeometry, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\nexport const CACHE_PAGE_GEOMETRY = 'CACHE_PAGE_GEOMETRY';\nexport const SET_SELECTION = 'SET_SELECTION';\nexport const START_SELECTION = 'START_SELECTION';\nexport const END_SELECTION = 'END_SELECTION';\nexport const CLEAR_SELECTION = 'CLEAR_SELECTION';\nexport const SET_RECTS = 'SET_RECTS';\nexport const SET_SLICES = 'SET_SLICES';\nexport const RESET = 'RESET';\n\nexport interface CachePageGeometryAction extends Action {\n type: typeof CACHE_PAGE_GEOMETRY;\n payload: { page: number; geo: PdfPageGeometry };\n}\nexport interface SetSelectionAction extends Action {\n type: typeof SET_SELECTION;\n payload: SelectionRangeX | null;\n}\n\nexport interface StartSelectionAction extends Action {\n type: typeof START_SELECTION;\n}\n\nexport interface EndSelectionAction extends Action {\n type: typeof END_SELECTION;\n}\n\nexport interface ClearSelectionAction extends Action {\n type: typeof CLEAR_SELECTION;\n}\n\nexport interface SetRectsAction extends Action {\n type: typeof SET_RECTS;\n payload: Record<number, Rect[]>;\n}\n\nexport interface SetSlicesAction extends Action {\n type: typeof SET_SLICES;\n payload: Record<number, { start: number; count: number }>;\n}\n\nexport interface ResetAction extends Action {\n type: typeof RESET;\n}\n\nexport type SelectionAction =\n | CachePageGeometryAction\n | SetSelectionAction\n | StartSelectionAction\n | EndSelectionAction\n | ClearSelectionAction\n | SetRectsAction\n | SetSlicesAction\n | ResetAction;\n\nexport const cachePageGeometry = (page: number, geo: PdfPageGeometry): CachePageGeometryAction => ({\n type: CACHE_PAGE_GEOMETRY,\n payload: { page, geo },\n});\n\nexport const setSelection = (sel: SelectionRangeX): SetSelectionAction => ({\n type: SET_SELECTION,\n payload: sel,\n});\n\nexport const startSelection = (): StartSelectionAction => ({ type: START_SELECTION });\n\nexport const endSelection = (): EndSelectionAction => ({ type: END_SELECTION });\n\nexport const clearSelection = (): ClearSelectionAction => ({ type: CLEAR_SELECTION });\n\nexport const setRects = (allRects: Record<number, Rect[]>): SetRectsAction => ({\n type: SET_RECTS,\n payload: allRects,\n});\n\nexport const setSlices = (\n slices: Record<number, { start: number; count: number }>,\n): SetSlicesAction => ({ type: SET_SLICES, payload: slices });\n\nexport const reset = (): ResetAction => ({ type: RESET });\n","import { Rect, boundingRect } from '@embedpdf/models';\nimport { FormattedSelection, SelectionState } from './types';\n\nexport function selectRectsForPage(state: SelectionState, page: number) {\n return state.rects[page] ?? [];\n}\n\nexport function selectBoundingRectForPage(state: SelectionState, page: number) {\n return boundingRect(selectRectsForPage(state, page));\n}\n\nexport function selectRectsAndBoundingRectForPage(state: SelectionState, page: number) {\n return {\n rects: selectRectsForPage(state, page),\n boundingRect: selectBoundingRectForPage(state, page),\n };\n}\n\nexport function selectBoundingRectsForAllPages(state: SelectionState) {\n const out: { page: number; rect: Rect }[] = [];\n const rectMap = state.rects;\n\n for (const key in rectMap) {\n const page = Number(key);\n const bRect = boundingRect(rectMap[page]);\n if (bRect) out.push({ page, rect: bRect });\n }\n return out;\n}\n\nexport function getFormattedSelectionForPage(\n state: SelectionState,\n page: number,\n): FormattedSelection | null {\n const segmentRects = state.rects[page] || [];\n if (segmentRects.length === 0) return null;\n const boundingRect = selectBoundingRectForPage(state, page);\n if (!boundingRect) return null;\n return { pageIndex: page, rect: boundingRect, segmentRects };\n}\n\nexport function getFormattedSelection(state: SelectionState) {\n const result: FormattedSelection[] = [];\n\n // Get all pages that have rects\n const pages = Object.keys(state.rects).map(Number);\n\n for (const pageIndex of pages) {\n const segmentRects = state.rects[pageIndex] || [];\n\n if (segmentRects.length === 0) continue;\n\n // Calculate bounding rect for this page\n const boundingRect = selectBoundingRectForPage(state, pageIndex);\n\n if (boundingRect) {\n result.push({\n pageIndex,\n rect: boundingRect,\n segmentRects,\n });\n }\n }\n\n return result;\n}\n","import { PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport { SelectionRangeX } from './types';\n\n/**\n * Hit-test helper using runs\n * @param geo - page geometry\n * @param pt - point\n * @returns glyph index\n */\nexport function glyphAt(geo: PdfPageGeometry, pt: Position) {\n for (const run of geo.runs) {\n const inRun =\n pt.y >= run.rect.y &&\n pt.y <= run.rect.y + run.rect.height &&\n pt.x >= run.rect.x &&\n pt.x <= run.rect.x + run.rect.width;\n\n if (!inRun) continue;\n\n // Simply check if the point is within any glyph's bounding box\n const rel = run.glyphs.findIndex(\n (g) => pt.x >= g.x && pt.x <= g.x + g.width && pt.y >= g.y && pt.y <= g.y + g.height,\n );\n\n if (rel !== -1) {\n return run.charStart + rel;\n }\n }\n return -1;\n}\n\n/**\n * Helper: min/max glyph indices on `page` for current sel\n * @param sel - selection range\n * @param geo - page geometry\n * @param page - page index\n * @returns { from: number; to: number } | null\n */\nexport function sliceBounds(\n sel: SelectionRangeX | null,\n geo: PdfPageGeometry | undefined,\n page: number,\n): { from: number; to: number } | null {\n if (!sel || !geo) return null;\n if (page < sel.start.page || page > sel.end.page) return null;\n\n const from = page === sel.start.page ? sel.start.index : 0;\n\n const lastRun = geo.runs[geo.runs.length - 1];\n const lastCharOnPage = lastRun.charStart + lastRun.glyphs.length - 1;\n\n const to = page === sel.end.page ? sel.end.index : lastCharOnPage;\n\n return { from, to };\n}\n\n/**\n * Helper: build rects for a slice of the page\n * @param geo - page geometry\n * @param from - from index\n * @param to - to index\n * @param merge - whether to merge adjacent rects (default: true)\n * @returns rects\n */\nexport function rectsWithinSlice(\n geo: PdfPageGeometry,\n from: number,\n to: number,\n merge: boolean = true,\n): Rect[] {\n const textRuns: TextRunInfo[] = [];\n\n for (const run of geo.runs) {\n const runStart = run.charStart;\n const runEnd = runStart + run.glyphs.length - 1;\n if (runEnd < from || runStart > to) continue;\n\n const sIdx = Math.max(from, runStart) - runStart;\n const eIdx = Math.min(to, runEnd) - runStart;\n\n let minX = Infinity,\n maxX = -Infinity;\n let minY = Infinity,\n maxY = -Infinity;\n let charCount = 0;\n\n for (let i = sIdx; i <= eIdx; i++) {\n const g = run.glyphs[i];\n if (g.flags === 2) continue; // empty glyph\n\n minX = Math.min(minX, g.x);\n maxX = Math.max(maxX, g.x + g.width);\n minY = Math.min(minY, g.y);\n maxY = Math.max(maxY, g.y + g.height);\n charCount++;\n }\n\n if (minX !== Infinity && charCount > 0) {\n textRuns.push({\n rect: {\n origin: { x: minX, y: minY },\n size: { width: maxX - minX, height: maxY - minY },\n },\n charCount,\n });\n }\n }\n\n // If merge is false, just return the individual rects\n if (!merge) {\n return textRuns.map((run) => run.rect);\n }\n\n // Otherwise merge adjacent rects\n return mergeAdjacentRects(textRuns);\n}\n\n/**\n * ============================================================================\n * Rectangle Merging Algorithm\n * ============================================================================\n *\n * The following code is adapted from Chromium's PDF text selection implementation.\n *\n * Copyright 2010 The Chromium Authors\n * Use of this source code is governed by a BSD-style license that can be\n * found in the LICENSE file: https://source.chromium.org/chromium/chromium/src/+/main:LICENSE\n *\n * Original source:\n * https://source.chromium.org/chromium/chromium/src/+/main:pdf/pdfium/pdfium_range.cc\n *\n * Adapted for TypeScript and this project's Rect/geometry types.\n */\n\n/**\n * Text run info for rect merging (similar to Chromium's ScreenRectTextRunInfo)\n */\nexport interface TextRunInfo {\n rect: Rect;\n charCount: number;\n}\n\n/**\n * Helper functions for Rect operations\n */\nexport function rectUnion(rect1: Rect, rect2: Rect): Rect {\n const left = Math.min(rect1.origin.x, rect2.origin.x);\n const top = Math.min(rect1.origin.y, rect2.origin.y);\n const right = Math.max(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.max(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n return {\n origin: { x: left, y: top },\n size: { width: right - left, height: bottom - top },\n };\n}\n\nexport function rectIntersect(rect1: Rect, rect2: Rect): Rect {\n const left = Math.max(rect1.origin.x, rect2.origin.x);\n const top = Math.max(rect1.origin.y, rect2.origin.y);\n const right = Math.min(rect1.origin.x + rect1.size.width, rect2.origin.x + rect2.size.width);\n const bottom = Math.min(rect1.origin.y + rect1.size.height, rect2.origin.y + rect2.size.height);\n\n const width = Math.max(0, right - left);\n const height = Math.max(0, bottom - top);\n\n return {\n origin: { x: left, y: top },\n size: { width, height },\n };\n}\n\nexport function rectIsEmpty(rect: Rect): boolean {\n return rect.size.width <= 0 || rect.size.height <= 0;\n}\n\n/**\n * Returns a ratio between [0, 1] representing vertical overlap\n */\nexport function getVerticalOverlap(rect1: Rect, rect2: Rect): number {\n if (rectIsEmpty(rect1) || rectIsEmpty(rect2)) return 0;\n\n const unionRect = rectUnion(rect1, rect2);\n\n if (unionRect.size.height === rect1.size.height || unionRect.size.height === rect2.size.height) {\n return 1.0;\n }\n\n const intersectRect = rectIntersect(rect1, rect2);\n return intersectRect.size.height / unionRect.size.height;\n}\n\n/**\n * Returns true if there is sufficient horizontal and vertical overlap\n */\nexport function shouldMergeHorizontalRects(textRun1: TextRunInfo, textRun2: TextRunInfo): boolean {\n const VERTICAL_OVERLAP_THRESHOLD = 0.8;\n const rect1 = textRun1.rect;\n const rect2 = textRun2.rect;\n\n if (getVerticalOverlap(rect1, rect2) < VERTICAL_OVERLAP_THRESHOLD) {\n return false;\n }\n\n const HORIZONTAL_WIDTH_FACTOR = 1.0;\n const averageWidth1 = (HORIZONTAL_WIDTH_FACTOR * rect1.size.width) / textRun1.charCount;\n const averageWidth2 = (HORIZONTAL_WIDTH_FACTOR * rect2.size.width) / textRun2.charCount;\n\n const rect1Left = rect1.origin.x - averageWidth1;\n const rect1Right = rect1.origin.x + rect1.size.width + averageWidth1;\n const rect2Left = rect2.origin.x - averageWidth2;\n const rect2Right = rect2.origin.x + rect2.size.width + averageWidth2;\n\n return rect1Left < rect2Right && rect1Right > rect2Left;\n}\n\n/**\n * Merge adjacent rectangles based on proximity and overlap (similar to Chromium's algorithm)\n */\nexport function mergeAdjacentRects(textRuns: TextRunInfo[]): Rect[] {\n const results: Rect[] = [];\n let previousTextRun: TextRunInfo | null = null;\n let currentRect: Rect | null = null;\n\n for (const textRun of textRuns) {\n if (previousTextRun && currentRect) {\n if (shouldMergeHorizontalRects(previousTextRun, textRun)) {\n currentRect = rectUnion(currentRect, textRun.rect);\n } else {\n results.push(currentRect);\n currentRect = textRun.rect;\n }\n } else {\n currentRect = textRun.rect;\n }\n previousTextRun = textRun;\n }\n\n if (currentRect && !rectIsEmpty(currentRect)) {\n results.push(currentRect);\n }\n\n return results;\n}\n","import {\n BasePlugin,\n PluginRegistry,\n REFRESH_PAGES,\n SET_DOCUMENT,\n Unsubscribe,\n createBehaviorEmitter,\n createEmitter,\n} from '@embedpdf/core';\nimport {\n PdfEngine,\n PdfPageGeometry,\n Rect,\n PdfTask,\n PdfTaskHelper,\n PdfErrorCode,\n ignore,\n PageTextSlice,\n Task,\n} from '@embedpdf/models';\n\nimport {\n cachePageGeometry,\n setSelection,\n SelectionAction,\n endSelection,\n startSelection,\n clearSelection,\n reset,\n setRects,\n setSlices,\n} from './actions';\nimport * as selector from './selectors';\nimport {\n SelectionCapability,\n SelectionPluginConfig,\n SelectionRangeX,\n SelectionState,\n} from './types';\nimport { sliceBounds, rectsWithinSlice } from './utils';\n\nexport class SelectionPlugin extends BasePlugin<\n SelectionPluginConfig,\n SelectionCapability,\n SelectionState,\n SelectionAction\n> {\n static readonly id = 'selection' as const;\n\n /** Modes that should trigger text-selection logic */\n private enabledModes = new Set<string>(['pointerMode']);\n\n /* interactive state */\n private selecting = false;\n private anchor?: { page: number; index: number };\n\n private readonly selChange$ = createBehaviorEmitter<SelectionState['selection']>();\n private readonly textRetrieved$ = createBehaviorEmitter<string[]>();\n private readonly copyToClipboard$ = createEmitter<string>();\n private readonly beginSelection$ = createEmitter<{ page: number; index: number }>();\n private readonly endSelection$ = createEmitter<void>();\n private readonly refreshPages$ = createEmitter<number[]>();\n\n constructor(id: string, registry: PluginRegistry) {\n super(id, registry);\n\n this.coreStore.onAction(SET_DOCUMENT, (_action) => {\n this.dispatch(reset());\n });\n\n this.coreStore.onAction(REFRESH_PAGES, (action) => {\n const tasks = action.payload.map((pageIdx) => this.getNewPageGeometryAndCache(pageIdx));\n Task.all(tasks).wait(() => {\n this.refreshPages$.emit(action.payload);\n }, ignore);\n });\n }\n\n /* ── life-cycle ────────────────────────────────────────── */\n async initialize() {}\n async destroy() {\n this.selChange$.clear();\n }\n\n /* ── capability exposed to UI / other plugins ─────────── */\n buildCapability(): SelectionCapability {\n return {\n getGeometry: (p) => this.getOrLoadGeometry(p),\n getFormattedSelection: () => selector.getFormattedSelection(this.state),\n getFormattedSelectionForPage: (p) => selector.getFormattedSelectionForPage(this.state, p),\n getHighlightRectsForPage: (p) => selector.selectRectsForPage(this.state, p),\n getHighlightRects: () => this.state.rects,\n getBoundingRectForPage: (p) => selector.selectBoundingRectForPage(this.state, p),\n getBoundingRects: () => selector.selectBoundingRectsForAllPages(this.state),\n begin: (p, i) => this.beginSelection(p, i),\n update: (p, i) => this.updateSelection(p, i),\n end: () => this.endSelection(),\n clear: () => this.clearSelection(),\n onCopyToClipboard: this.copyToClipboard$.on,\n onSelectionChange: this.selChange$.on,\n onTextRetrieved: this.textRetrieved$.on,\n onBeginSelection: this.beginSelection$.on,\n onEndSelection: this.endSelection$.on,\n getSelectedText: () => this.getSelectedText(),\n copyToClipboard: () => this.copyToClipboard(),\n enableForMode: (id: string) => this.enabledModes.add(id),\n isEnabledForMode: (id: string) => this.enabledModes.has(id),\n getState: () => this.state,\n };\n }\n\n public onRefreshPages(fn: (pages: number[]) => void): Unsubscribe {\n return this.refreshPages$.on(fn);\n }\n\n private getNewPageGeometryAndCache(pageIdx: number): PdfTask<PdfPageGeometry> {\n if (!this.coreState.core.document)\n return PdfTaskHelper.reject({ code: PdfErrorCode.NotFound, message: 'Doc Not Found' });\n const page = this.coreState.core.document.pages.find((p) => p.index === pageIdx)!;\n const task = this.engine.getPageGeometry(this.coreState.core.document, page);\n task.wait((geo) => {\n this.dispatch(cachePageGeometry(pageIdx, geo));\n }, ignore);\n return task;\n }\n\n /* ── geometry cache ───────────────────────────────────── */\n private getOrLoadGeometry(pageIdx: number): PdfTask<PdfPageGeometry> {\n const cached = this.state.geometry[pageIdx];\n if (cached) return PdfTaskHelper.resolve(cached);\n\n return this.getNewPageGeometryAndCache(pageIdx);\n }\n\n /* ── selection state updates ───────────────────────────── */\n private beginSelection(page: number, index: number) {\n this.selecting = true;\n this.anchor = { page, index };\n this.dispatch(startSelection());\n this.beginSelection$.emit({ page, index });\n }\n\n private endSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(endSelection());\n this.endSelection$.emit();\n }\n\n private clearSelection() {\n this.selecting = false;\n this.anchor = undefined;\n this.dispatch(clearSelection());\n this.selChange$.emit(null);\n }\n\n private updateSelection(page: number, index: number) {\n if (!this.selecting || !this.anchor) return;\n\n const a = this.anchor;\n const forward = page > a.page || (page === a.page && index >= a.index);\n\n const start = forward ? a : { page, index };\n const end = forward ? { page, index } : a;\n\n const range = { start, end };\n this.dispatch(setSelection(range));\n this.updateRectsAndSlices(range);\n this.selChange$.emit(range);\n }\n\n private updateRectsAndSlices(range: SelectionRangeX) {\n const allRects: Record<number, Rect[]> = {};\n const allSlices: Record<number, { start: number; count: number }> = {};\n\n for (let p = range.start.page; p <= range.end.page; p++) {\n const geo = this.state.geometry[p];\n const sb = sliceBounds(range, geo, p);\n if (!sb) continue;\n\n allRects[p] = rectsWithinSlice(geo!, sb.from, sb.to);\n allSlices[p] = { start: sb.from, count: sb.to - sb.from + 1 };\n }\n\n this.dispatch(setRects(allRects));\n this.dispatch(setSlices(allSlices));\n }\n\n private getSelectedText(): PdfTask<string[]> {\n if (!this.coreState.core.document || !this.state.selection) {\n return PdfTaskHelper.reject({\n code: PdfErrorCode.NotFound,\n message: 'Doc Not Found or No Selection',\n });\n }\n\n const sel = this.state.selection;\n const req: PageTextSlice[] = [];\n\n for (let p = sel.start.page; p <= sel.end.page; p++) {\n const s = this.state.slices[p];\n if (s) req.push({ pageIndex: p, charIndex: s.start, charCount: s.count });\n }\n\n if (req.length === 0) return PdfTaskHelper.resolve([] as string[]);\n\n const task = this.engine.getTextSlices(this.coreState.core.document, req);\n\n // Emit the text when it's retrieved\n task.wait((text) => {\n this.textRetrieved$.emit(text);\n }, ignore);\n\n return task;\n }\n\n private copyToClipboard() {\n const text = this.getSelectedText();\n text.wait((text) => {\n this.copyToClipboard$.emit(text.join('\\n'));\n }, ignore);\n }\n}\n","import { SelectionState } from './types';\nimport {\n SelectionAction,\n CACHE_PAGE_GEOMETRY,\n SET_SELECTION,\n START_SELECTION,\n END_SELECTION,\n CLEAR_SELECTION,\n RESET,\n SET_SLICES,\n SET_RECTS,\n} from './actions';\n\nexport const initialState: SelectionState = {\n geometry: {},\n rects: {},\n slices: {},\n selection: null,\n active: false,\n selecting: false,\n};\n\nexport const selectionReducer = (state = initialState, action: SelectionAction): SelectionState => {\n switch (action.type) {\n case CACHE_PAGE_GEOMETRY: {\n const { page, geo } = action.payload;\n return { ...state, geometry: { ...state.geometry, [page]: geo } };\n }\n case SET_SELECTION:\n return { ...state, selection: action.payload, active: true };\n case START_SELECTION:\n return { ...state, selecting: true, selection: null, rects: {} };\n case END_SELECTION:\n return { ...state, selecting: false };\n case CLEAR_SELECTION:\n return { ...state, selecting: false, selection: null, rects: {}, active: false };\n case SET_RECTS:\n return { ...state, rects: action.payload };\n case SET_SLICES:\n return { ...state, slices: action.payload };\n case RESET:\n return initialState;\n default:\n return state;\n }\n};\n","import { PluginPackage } from '@embedpdf/core';\nimport { manifest, SELECTION_PLUGIN_ID } from './manifest';\nimport { SelectionPluginConfig, SelectionState } from './types';\n\nimport { SelectionPlugin } from './selection-plugin';\nimport { SelectionAction } from './actions';\nimport { selectionReducer, initialState } from './reducer';\n\nexport const SelectionPluginPackage: PluginPackage<\n SelectionPlugin,\n SelectionPluginConfig,\n SelectionState,\n SelectionAction\n> = {\n manifest,\n create: (registry) => new SelectionPlugin(SELECTION_PLUGIN_ID, registry),\n reducer: selectionReducer,\n initialState,\n};\n\nexport * from './selection-plugin';\nexport * from './types';\nexport * from './manifest';\nexport * from './utils';\n"],"names":["boundingRect","selector.getFormattedSelection","selector.getFormattedSelectionForPage","selector.selectRectsForPage","selector.selectBoundingRectForPage","selector.selectBoundingRectsForAllPages","text"],"mappings":"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"}
@@ -1,9 +1,7 @@
1
- import { BasePlugin, PluginRegistry } from '@embedpdf/core';
2
- import { PdfEngine } from '@embedpdf/models';
1
+ import { BasePlugin, PluginRegistry, Unsubscribe } from '@embedpdf/core';
3
2
  import { SelectionAction } from './actions';
4
3
  import { SelectionCapability, SelectionPluginConfig, SelectionState } from './types';
5
4
  export declare class SelectionPlugin extends BasePlugin<SelectionPluginConfig, SelectionCapability, SelectionState, SelectionAction> {
6
- private engine;
7
5
  static readonly id: "selection";
8
6
  /** Modes that should trigger text-selection logic */
9
7
  private enabledModes;
@@ -14,10 +12,13 @@ export declare class SelectionPlugin extends BasePlugin<SelectionPluginConfig, S
14
12
  private readonly copyToClipboard$;
15
13
  private readonly beginSelection$;
16
14
  private readonly endSelection$;
17
- constructor(id: string, registry: PluginRegistry, engine: PdfEngine);
15
+ private readonly refreshPages$;
16
+ constructor(id: string, registry: PluginRegistry);
18
17
  initialize(): Promise<void>;
19
18
  destroy(): Promise<void>;
20
19
  buildCapability(): SelectionCapability;
20
+ onRefreshPages(fn: (pages: number[]) => void): Unsubscribe;
21
+ private getNewPageGeometryAndCache;
21
22
  private getOrLoadGeometry;
22
23
  private beginSelection;
23
24
  private endSelection;
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("@embedpdf/core/preact"),t=require("@embedpdf/plugin-selection"),r=require("preact/jsx-runtime");require("preact");const i=require("preact/hooks"),o=require("@embedpdf/models"),n=require("@embedpdf/plugin-interaction-manager/preact"),s=()=>e.useCapability(t.SelectionPlugin.id);exports.CopyToClipboard=function(){const{provides:e}=s();return i.useEffect((()=>{if(e)return e.onCopyToClipboard((e=>{navigator.clipboard.writeText(e)}))}),[e]),null},exports.SelectionLayer=function({pageIndex:e,scale:l,background:u="rgba(33,150,243)"}){const{provides:a}=s(),{provides:d}=n.useInteractionManagerCapability(),{register:c}=n.usePointerHandlers({pageIndex:e}),[g,p]=i.useState([]),[f,b]=i.useState(null),{setCursor:h,removeCursor:x}=n.useCursor(),y=i.useRef(null);i.useEffect((()=>{if(a)return a.onSelectionChange((()=>{"pointerMode"===(null==d?void 0:d.getActiveMode())?(p(a.getHighlightRectsForPage(e)),b(a.getBoundingRectForPage(e))):(p([]),b(null))}))}),[a,e]);const m=i.useCallback((e=>{const r=y.current;return r?t.glyphAt(r,e):-1}),[]);i.useEffect((()=>{if(!a)return;const t=a.getGeometry(e);return t.wait((e=>y.current=e),o.ignore),()=>{t.abort({code:o.PdfErrorCode.Cancelled,message:"Cancelled"}),y.current=null}}),[a,e]);const C=i.useMemo((()=>({onPointerDown:(r,i,n)=>{if(!a)return;if(!a.isEnabledForMode(n))return;a.clear();a.getGeometry(e).wait((i=>{const o=t.glyphAt(i,r);-1!==o&&a.begin(e,o)}),o.ignore)},onPointerMove:(t,r,i)=>{if(!a)return;if(!a.isEnabledForMode(i))return;const o=m(t);-1!==o?h("selection-text","text",10):x("selection-text"),-1!==o&&a.update(e,o)},onPointerUp:(e,t,r)=>{a&&a.isEnabledForMode(r)&&a.end()},onHandlerActiveEnd(e){a&&a.isEnabledForMode(e)&&a.clear()}})),[a,e,m]);return i.useEffect((()=>{if(c)return c(C)}),[c,C]),f?r.jsx("div",{style:{position:"absolute",left:f.origin.x*l,top:f.origin.y*l,width:f.size.width*l,height:f.size.height*l,mixBlendMode:"multiply",isolation:"isolate"},children:g.map(((e,t)=>r.jsx("div",{style:{position:"absolute",left:(e.origin.x-f.origin.x)*l,top:(e.origin.y-f.origin.y)*l,width:e.size.width*l,height:e.size.height*l,background:u,pointerEvents:"none"}},t)))}):null},exports.useSelectionCapability=s,exports.useSelectionPlugin=()=>e.usePlugin(t.SelectionPlugin.id);
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("@embedpdf/core/preact"),t=require("@embedpdf/plugin-selection"),r=require("preact/jsx-runtime");require("preact");const i=require("preact/hooks"),n=require("@embedpdf/models"),o=require("@embedpdf/plugin-interaction-manager/preact"),s=()=>e.useCapability(t.SelectionPlugin.id),u=()=>e.usePlugin(t.SelectionPlugin.id);exports.CopyToClipboard=function(){const{provides:e}=s();return i.useEffect((()=>{if(e)return e.onCopyToClipboard((e=>{navigator.clipboard.writeText(e)}))}),[e]),null},exports.SelectionLayer=function({pageIndex:e,scale:l,background:a="rgba(33,150,243)"}){const{provides:c}=s(),{plugin:d}=u(),{provides:g}=o.useInteractionManagerCapability(),{register:p}=o.usePointerHandlers({pageIndex:e}),[f,b]=i.useState([]),[h,y]=i.useState(null),{setCursor:x,removeCursor:m}=o.useCursor(),C=i.useRef(null);i.useEffect((()=>{if(c)return c.onSelectionChange((()=>{"pointerMode"===(null==g?void 0:g.getActiveMode())?(b(c.getHighlightRectsForPage(e)),y(c.getBoundingRectForPage(e))):(b([]),y(null))}))}),[c,e]);const v=i.useCallback((e=>{const r=C.current;return r?t.glyphAt(r,e):-1}),[]);i.useEffect((()=>{if(!c)return;const t=c.getGeometry(e);return t.wait((e=>C.current=e),n.ignore),()=>{t.abort({code:n.PdfErrorCode.Cancelled,message:"Cancelled"}),C.current=null}}),[c,e]),i.useEffect((()=>{if(d&&c)return d.onRefreshPages((t=>{if(t.includes(e)){c.getGeometry(e).wait((e=>C.current=e),n.ignore)}}))}),[c,d,e]);const P=i.useMemo((()=>({onPointerDown:(r,i,o)=>{if(!c)return;if(!c.isEnabledForMode(o))return;c.clear();c.getGeometry(e).wait((i=>{const n=t.glyphAt(i,r);-1!==n&&c.begin(e,n)}),n.ignore)},onPointerMove:(t,r,i)=>{if(!c)return;if(!c.isEnabledForMode(i))return;const n=v(t);-1!==n?x("selection-text","text",10):m("selection-text"),-1!==n&&c.update(e,n)},onPointerUp:(e,t,r)=>{c&&c.isEnabledForMode(r)&&c.end()},onHandlerActiveEnd(e){c&&c.isEnabledForMode(e)&&c.clear()}})),[c,e,v]);return i.useEffect((()=>{if(p)return p(P)}),[p,P]),h?r.jsx("div",{style:{position:"absolute",left:h.origin.x*l,top:h.origin.y*l,width:h.size.width*l,height:h.size.height*l,mixBlendMode:"multiply",isolation:"isolate"},children:f.map(((e,t)=>r.jsx("div",{style:{position:"absolute",left:(e.origin.x-h.origin.x)*l,top:(e.origin.y-h.origin.y)*l,width:e.size.width*l,height:e.size.height*l,background:a,pointerEvents:"none"}},t)))}):null},exports.useSelectionCapability=s,exports.useSelectionPlugin=u;
2
2
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/copy-to-clipboard.tsx","../../src/shared/components/selection-layer.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n"],"names":["useSelectionCapability","useCapability","SelectionPlugin","id","provides","sel","useEffect","onCopyToClipboard","text","navigator","clipboard","writeText","pageIndex","scale","background","im","useInteractionManagerCapability","register","usePointerHandlers","rects","setRects","useState","boundingRect","setBoundingRect","setCursor","removeCursor","useCursor","geoCacheRef","useRef","onSelectionChange","getActiveMode","getHighlightRectsForPage","getBoundingRectForPage","cachedGlyphAt","useCallback","pt","geo","current","glyphAt","task","getGeometry","wait","g","ignore","abort","code","PdfErrorCode","Cancelled","message","handlers","useMemo","onPointerDown","point","_evt","modeId","isEnabledForMode","clear","begin","onPointerMove","update","onPointerUp","_point","end","onHandlerActiveEnd","jsxRuntime","jsx","style","position","left","origin","x","top","y","width","size","height","mixBlendMode","isolation","children","map","b","i","pointerEvents","usePlugin"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/copy-to-clipboard.tsx","../../src/shared/components/selection-layer.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability, useSelectionPlugin } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { plugin: selectionPlugin } = useSelectionPlugin();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n useEffect(() => {\n if (!selectionPlugin || !sel) return;\n return selectionPlugin.onRefreshPages((pages) => {\n if (pages.includes(pageIndex)) {\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n }\n });\n }, [sel, selectionPlugin, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n"],"names":["useSelectionCapability","useCapability","SelectionPlugin","id","useSelectionPlugin","usePlugin","provides","sel","useEffect","onCopyToClipboard","text","navigator","clipboard","writeText","pageIndex","scale","background","plugin","selectionPlugin","im","useInteractionManagerCapability","register","usePointerHandlers","rects","setRects","useState","boundingRect","setBoundingRect","setCursor","removeCursor","useCursor","geoCacheRef","useRef","onSelectionChange","getActiveMode","getHighlightRectsForPage","getBoundingRectForPage","cachedGlyphAt","useCallback","pt","geo","current","glyphAt","task","getGeometry","wait","g","ignore","abort","code","PdfErrorCode","Cancelled","message","onRefreshPages","pages","includes","handlers","useMemo","onPointerDown","point","_evt","modeId","isEnabledForMode","clear","begin","onPointerMove","update","onPointerUp","_point","end","onHandlerActiveEnd","jsxRuntime","jsx","style","position","left","origin","x","top","y","width","size","height","mixBlendMode","isolation","children","map","b","i","pointerEvents"],"mappings":"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"}
@@ -9,6 +9,7 @@ const useSelectionCapability = () => useCapability(SelectionPlugin.id);
9
9
  const useSelectionPlugin = () => usePlugin(SelectionPlugin.id);
10
10
  function SelectionLayer({ pageIndex, scale, background = "rgba(33,150,243)" }) {
11
11
  const { provides: sel } = useSelectionCapability();
12
+ const { plugin: selectionPlugin } = useSelectionPlugin();
12
13
  const { provides: im } = useInteractionManagerCapability();
13
14
  const { register } = usePointerHandlers({ pageIndex });
14
15
  const [rects, setRects] = useState([]);
@@ -44,6 +45,15 @@ function SelectionLayer({ pageIndex, scale, background = "rgba(33,150,243)" }) {
44
45
  geoCacheRef.current = null;
45
46
  };
46
47
  }, [sel, pageIndex]);
48
+ useEffect(() => {
49
+ if (!selectionPlugin || !sel) return;
50
+ return selectionPlugin.onRefreshPages((pages) => {
51
+ if (pages.includes(pageIndex)) {
52
+ const task = sel.getGeometry(pageIndex);
53
+ task.wait((g) => geoCacheRef.current = g, ignore);
54
+ }
55
+ });
56
+ }, [sel, selectionPlugin, pageIndex]);
47
57
  const handlers = useMemo(
48
58
  () => ({
49
59
  onPointerDown: (point, _evt, modeId) => {
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/selection-layer.tsx","../../src/shared/components/copy-to-clipboard.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/selection-layer.tsx","../../src/shared/components/copy-to-clipboard.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability, useSelectionPlugin } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { plugin: selectionPlugin } = useSelectionPlugin();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n useEffect(() => {\n if (!selectionPlugin || !sel) return;\n return selectionPlugin.onRefreshPages((pages) => {\n if (pages.includes(pageIndex)) {\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n }\n });\n }, [sel, selectionPlugin, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n"],"names":[],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("@embedpdf/core/react"),t=require("@embedpdf/plugin-selection"),r=require("react/jsx-runtime"),i=require("react"),n=require("@embedpdf/models"),o=require("@embedpdf/plugin-interaction-manager/react"),l=()=>e.useCapability(t.SelectionPlugin.id);exports.CopyToClipboard=function(){const{provides:e}=l();return i.useEffect((()=>{if(e)return e.onCopyToClipboard((e=>{navigator.clipboard.writeText(e)}))}),[e]),null},exports.SelectionLayer=function({pageIndex:e,scale:s,background:u="rgba(33,150,243)"}){const{provides:a}=l(),{provides:d}=o.useInteractionManagerCapability(),{register:c}=o.usePointerHandlers({pageIndex:e}),[g,p]=i.useState([]),[f,b]=i.useState(null),{setCursor:h,removeCursor:x}=o.useCursor(),y=i.useRef(null);i.useEffect((()=>{if(a)return a.onSelectionChange((()=>{"pointerMode"===(null==d?void 0:d.getActiveMode())?(p(a.getHighlightRectsForPage(e)),b(a.getBoundingRectForPage(e))):(p([]),b(null))}))}),[a,e]);const m=i.useCallback((e=>{const r=y.current;return r?t.glyphAt(r,e):-1}),[]);i.useEffect((()=>{if(!a)return;const t=a.getGeometry(e);return t.wait((e=>y.current=e),n.ignore),()=>{t.abort({code:n.PdfErrorCode.Cancelled,message:"Cancelled"}),y.current=null}}),[a,e]);const C=i.useMemo((()=>({onPointerDown:(r,i,o)=>{if(!a)return;if(!a.isEnabledForMode(o))return;a.clear();a.getGeometry(e).wait((i=>{const n=t.glyphAt(i,r);-1!==n&&a.begin(e,n)}),n.ignore)},onPointerMove:(t,r,i)=>{if(!a)return;if(!a.isEnabledForMode(i))return;const n=m(t);-1!==n?h("selection-text","text",10):x("selection-text"),-1!==n&&a.update(e,n)},onPointerUp:(e,t,r)=>{a&&a.isEnabledForMode(r)&&a.end()},onHandlerActiveEnd(e){a&&a.isEnabledForMode(e)&&a.clear()}})),[a,e,m]);return i.useEffect((()=>{if(c)return c(C)}),[c,C]),f?r.jsx("div",{style:{position:"absolute",left:f.origin.x*s,top:f.origin.y*s,width:f.size.width*s,height:f.size.height*s,mixBlendMode:"multiply",isolation:"isolate"},children:g.map(((e,t)=>r.jsx("div",{style:{position:"absolute",left:(e.origin.x-f.origin.x)*s,top:(e.origin.y-f.origin.y)*s,width:e.size.width*s,height:e.size.height*s,background:u,pointerEvents:"none"}},t)))}):null},exports.useSelectionCapability=l,exports.useSelectionPlugin=()=>e.usePlugin(t.SelectionPlugin.id);
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("@embedpdf/core/react"),t=require("@embedpdf/plugin-selection"),r=require("react/jsx-runtime"),i=require("react"),n=require("@embedpdf/models"),o=require("@embedpdf/plugin-interaction-manager/react"),s=()=>e.useCapability(t.SelectionPlugin.id),l=()=>e.usePlugin(t.SelectionPlugin.id);exports.CopyToClipboard=function(){const{provides:e}=s();return i.useEffect((()=>{if(e)return e.onCopyToClipboard((e=>{navigator.clipboard.writeText(e)}))}),[e]),null},exports.SelectionLayer=function({pageIndex:e,scale:u,background:a="rgba(33,150,243)"}){const{provides:d}=s(),{plugin:c}=l(),{provides:g}=o.useInteractionManagerCapability(),{register:p}=o.usePointerHandlers({pageIndex:e}),[f,b]=i.useState([]),[h,y]=i.useState(null),{setCursor:x,removeCursor:m}=o.useCursor(),C=i.useRef(null);i.useEffect((()=>{if(d)return d.onSelectionChange((()=>{"pointerMode"===(null==g?void 0:g.getActiveMode())?(b(d.getHighlightRectsForPage(e)),y(d.getBoundingRectForPage(e))):(b([]),y(null))}))}),[d,e]);const v=i.useCallback((e=>{const r=C.current;return r?t.glyphAt(r,e):-1}),[]);i.useEffect((()=>{if(!d)return;const t=d.getGeometry(e);return t.wait((e=>C.current=e),n.ignore),()=>{t.abort({code:n.PdfErrorCode.Cancelled,message:"Cancelled"}),C.current=null}}),[d,e]),i.useEffect((()=>{if(c&&d)return c.onRefreshPages((t=>{if(t.includes(e)){d.getGeometry(e).wait((e=>C.current=e),n.ignore)}}))}),[d,c,e]);const P=i.useMemo((()=>({onPointerDown:(r,i,o)=>{if(!d)return;if(!d.isEnabledForMode(o))return;d.clear();d.getGeometry(e).wait((i=>{const n=t.glyphAt(i,r);-1!==n&&d.begin(e,n)}),n.ignore)},onPointerMove:(t,r,i)=>{if(!d)return;if(!d.isEnabledForMode(i))return;const n=v(t);-1!==n?x("selection-text","text",10):m("selection-text"),-1!==n&&d.update(e,n)},onPointerUp:(e,t,r)=>{d&&d.isEnabledForMode(r)&&d.end()},onHandlerActiveEnd(e){d&&d.isEnabledForMode(e)&&d.clear()}})),[d,e,v]);return i.useEffect((()=>{if(p)return p(P)}),[p,P]),h?r.jsx("div",{style:{position:"absolute",left:h.origin.x*u,top:h.origin.y*u,width:h.size.width*u,height:h.size.height*u,mixBlendMode:"multiply",isolation:"isolate"},children:f.map(((e,t)=>r.jsx("div",{style:{position:"absolute",left:(e.origin.x-h.origin.x)*u,top:(e.origin.y-h.origin.y)*u,width:e.size.width*u,height:e.size.height*u,background:a,pointerEvents:"none"}},t)))}):null},exports.useSelectionCapability=s,exports.useSelectionPlugin=l;
2
2
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/copy-to-clipboard.tsx","../../src/shared/components/selection-layer.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n"],"names":["useSelectionCapability","useCapability","SelectionPlugin","id","provides","sel","useEffect","onCopyToClipboard","text","navigator","clipboard","writeText","pageIndex","scale","background","im","useInteractionManagerCapability","register","usePointerHandlers","rects","setRects","useState","boundingRect","setBoundingRect","setCursor","removeCursor","useCursor","geoCacheRef","useRef","onSelectionChange","getActiveMode","getHighlightRectsForPage","getBoundingRectForPage","cachedGlyphAt","useCallback","pt","geo","current","glyphAt","task","getGeometry","wait","g","ignore","abort","code","PdfErrorCode","Cancelled","message","handlers","useMemo","onPointerDown","point","_evt","modeId","isEnabledForMode","clear","begin","onPointerMove","update","onPointerUp","_point","end","onHandlerActiveEnd","jsxRuntime","jsx","style","position","left","origin","x","top","y","width","size","height","mixBlendMode","isolation","children","map","b","i","pointerEvents","usePlugin"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/copy-to-clipboard.tsx","../../src/shared/components/selection-layer.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability, useSelectionPlugin } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { plugin: selectionPlugin } = useSelectionPlugin();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n useEffect(() => {\n if (!selectionPlugin || !sel) return;\n return selectionPlugin.onRefreshPages((pages) => {\n if (pages.includes(pageIndex)) {\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n }\n });\n }, [sel, selectionPlugin, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n"],"names":["useSelectionCapability","useCapability","SelectionPlugin","id","useSelectionPlugin","usePlugin","provides","sel","useEffect","onCopyToClipboard","text","navigator","clipboard","writeText","pageIndex","scale","background","plugin","selectionPlugin","im","useInteractionManagerCapability","register","usePointerHandlers","rects","setRects","useState","boundingRect","setBoundingRect","setCursor","removeCursor","useCursor","geoCacheRef","useRef","onSelectionChange","getActiveMode","getHighlightRectsForPage","getBoundingRectForPage","cachedGlyphAt","useCallback","pt","geo","current","glyphAt","task","getGeometry","wait","g","ignore","abort","code","PdfErrorCode","Cancelled","message","onRefreshPages","pages","includes","handlers","useMemo","onPointerDown","point","_evt","modeId","isEnabledForMode","clear","begin","onPointerMove","update","onPointerUp","_point","end","onHandlerActiveEnd","jsxRuntime","jsx","style","position","left","origin","x","top","y","width","size","height","mixBlendMode","isolation","children","map","b","i","pointerEvents"],"mappings":"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"}
@@ -8,6 +8,7 @@ const useSelectionCapability = () => useCapability(SelectionPlugin.id);
8
8
  const useSelectionPlugin = () => usePlugin(SelectionPlugin.id);
9
9
  function SelectionLayer({ pageIndex, scale, background = "rgba(33,150,243)" }) {
10
10
  const { provides: sel } = useSelectionCapability();
11
+ const { plugin: selectionPlugin } = useSelectionPlugin();
11
12
  const { provides: im } = useInteractionManagerCapability();
12
13
  const { register } = usePointerHandlers({ pageIndex });
13
14
  const [rects, setRects] = useState([]);
@@ -43,6 +44,15 @@ function SelectionLayer({ pageIndex, scale, background = "rgba(33,150,243)" }) {
43
44
  geoCacheRef.current = null;
44
45
  };
45
46
  }, [sel, pageIndex]);
47
+ useEffect(() => {
48
+ if (!selectionPlugin || !sel) return;
49
+ return selectionPlugin.onRefreshPages((pages) => {
50
+ if (pages.includes(pageIndex)) {
51
+ const task = sel.getGeometry(pageIndex);
52
+ task.wait((g) => geoCacheRef.current = g, ignore);
53
+ }
54
+ });
55
+ }, [sel, selectionPlugin, pageIndex]);
46
56
  const handlers = useMemo(
47
57
  () => ({
48
58
  onPointerDown: (point, _evt, modeId) => {
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/selection-layer.tsx","../../src/shared/components/copy-to-clipboard.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../../src/shared/hooks/use-selection.ts","../../src/shared/components/selection-layer.tsx","../../src/shared/components/copy-to-clipboard.tsx"],"sourcesContent":["import { useCapability, usePlugin } from '@embedpdf/core/@framework';\nimport { SelectionPlugin } from '@embedpdf/plugin-selection';\n\nexport const useSelectionCapability = () => useCapability<SelectionPlugin>(SelectionPlugin.id);\nexport const useSelectionPlugin = () => usePlugin<SelectionPlugin>(SelectionPlugin.id);\n","import { useCallback, useEffect, useMemo, useRef, useState } from '@framework';\nimport { ignore, PdfErrorCode, PdfPageGeometry, Position, Rect } from '@embedpdf/models';\nimport {\n useCursor,\n useInteractionManagerCapability,\n usePointerHandlers,\n} from '@embedpdf/plugin-interaction-manager/@framework';\nimport { PointerEventHandlersWithLifecycle } from '@embedpdf/plugin-interaction-manager';\nimport { glyphAt } from '@embedpdf/plugin-selection';\n\nimport { useSelectionCapability, useSelectionPlugin } from '../hooks';\n\ntype Props = {\n pageIndex: number;\n scale: number;\n background?: string;\n};\n\nexport function SelectionLayer({ pageIndex, scale, background = 'rgba(33,150,243)' }: Props) {\n const { provides: sel } = useSelectionCapability();\n const { plugin: selectionPlugin } = useSelectionPlugin();\n const { provides: im } = useInteractionManagerCapability();\n const { register } = usePointerHandlers({ pageIndex });\n const [rects, setRects] = useState<Array<Rect>>([]);\n const [boundingRect, setBoundingRect] = useState<Rect | null>(null);\n const { setCursor, removeCursor } = useCursor();\n const geoCacheRef = useRef<PdfPageGeometry | null>(null);\n\n /* subscribe to rect updates */\n useEffect(() => {\n if (!sel) return;\n return sel.onSelectionChange(() => {\n const mode = im?.getActiveMode();\n if (mode === 'pointerMode') {\n setRects(sel.getHighlightRectsForPage(pageIndex));\n setBoundingRect(sel.getBoundingRectForPage(pageIndex));\n } else {\n setRects([]);\n setBoundingRect(null);\n }\n });\n }, [sel, pageIndex]);\n\n /* cheap glyphAt cache for the active page */\n const cachedGlyphAt = useCallback((pt: Position) => {\n const geo = geoCacheRef.current;\n return geo ? glyphAt(geo, pt) : -1;\n }, []);\n\n // Initialize geometry cache\n useEffect(() => {\n if (!sel) return;\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n\n return () => {\n task.abort({\n code: PdfErrorCode.Cancelled,\n message: 'Cancelled',\n });\n geoCacheRef.current = null;\n };\n }, [sel, pageIndex]);\n\n useEffect(() => {\n if (!selectionPlugin || !sel) return;\n return selectionPlugin.onRefreshPages((pages) => {\n if (pages.includes(pageIndex)) {\n const task = sel.getGeometry(pageIndex);\n task.wait((g) => (geoCacheRef.current = g), ignore);\n }\n });\n }, [sel, selectionPlugin, pageIndex]);\n\n const handlers = useMemo(\n (): PointerEventHandlersWithLifecycle<PointerEvent> => ({\n onPointerDown: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n // clear the selection\n sel.clear();\n const task = sel.getGeometry(pageIndex);\n task.wait((geo) => {\n const g = glyphAt(geo, point);\n if (g !== -1) sel.begin(pageIndex, g);\n }, ignore);\n },\n onPointerMove: (point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n const g = cachedGlyphAt(point);\n if (g !== -1) {\n setCursor('selection-text', 'text', 10);\n } else {\n removeCursor('selection-text');\n }\n if (g !== -1) sel.update(pageIndex, g);\n },\n onPointerUp: (_point, _evt, modeId) => {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n sel.end();\n },\n onHandlerActiveEnd(modeId) {\n if (!sel) return;\n if (!sel.isEnabledForMode(modeId)) return;\n\n sel.clear();\n },\n }),\n [sel, pageIndex, cachedGlyphAt],\n );\n\n useEffect(() => {\n if (!register) return;\n return register(handlers);\n }, [register, handlers]);\n\n if (!boundingRect) return null;\n\n return (\n <div\n style={{\n position: 'absolute',\n left: boundingRect.origin.x * scale,\n top: boundingRect.origin.y * scale,\n width: boundingRect.size.width * scale,\n height: boundingRect.size.height * scale,\n mixBlendMode: 'multiply',\n isolation: 'isolate',\n }}\n >\n {rects.map((b, i) => (\n <div\n key={i}\n style={{\n position: 'absolute',\n left: (b.origin.x - boundingRect.origin.x) * scale,\n top: (b.origin.y - boundingRect.origin.y) * scale,\n width: b.size.width * scale,\n height: b.size.height * scale,\n background,\n pointerEvents: 'none',\n }}\n />\n ))}\n </div>\n );\n}\n","import { useEffect } from '@framework';\n\nimport { useSelectionCapability } from '../hooks';\n\nexport function CopyToClipboard() {\n const { provides: sel } = useSelectionCapability();\n\n useEffect(() => {\n if (!sel) return;\n return sel.onCopyToClipboard((text) => {\n navigator.clipboard.writeText(text);\n });\n }, [sel]);\n\n return null;\n}\n"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@embedpdf/plugin-selection",
3
- "version": "1.0.17",
3
+ "version": "1.0.19",
4
4
  "type": "module",
5
5
  "main": "./dist/index.cjs",
6
6
  "module": "./dist/index.js",
@@ -28,24 +28,24 @@
28
28
  }
29
29
  },
30
30
  "dependencies": {
31
- "@embedpdf/models": "1.0.17"
31
+ "@embedpdf/models": "1.0.19"
32
32
  },
33
33
  "devDependencies": {
34
34
  "@types/react": "^18.2.0",
35
35
  "typescript": "^5.0.0",
36
36
  "@embedpdf/build": "1.0.0",
37
- "@embedpdf/core": "1.0.17",
38
- "@embedpdf/plugin-viewport": "1.0.17",
39
- "@embedpdf/plugin-interaction-manager": "1.0.17"
37
+ "@embedpdf/plugin-viewport": "1.0.19",
38
+ "@embedpdf/core": "1.0.19",
39
+ "@embedpdf/plugin-interaction-manager": "1.0.19"
40
40
  },
41
41
  "peerDependencies": {
42
42
  "react": ">=16.8.0",
43
43
  "react-dom": ">=16.8.0",
44
44
  "preact": "^10.26.4",
45
45
  "vue": ">=3.2.0",
46
- "@embedpdf/core": "1.0.17",
47
- "@embedpdf/plugin-interaction-manager": "1.0.17",
48
- "@embedpdf/plugin-viewport": "1.0.17"
46
+ "@embedpdf/core": "1.0.19",
47
+ "@embedpdf/plugin-viewport": "1.0.19",
48
+ "@embedpdf/plugin-interaction-manager": "1.0.19"
49
49
  },
50
50
  "files": [
51
51
  "dist",