lexical-ai-plugin
Version:
This plugin is built to be used with Cloudflare Workers. In order to use this plugin, deploy this [Code](https://github.com/akadotsh/lexical-ai-worker) to Cloudflare.
22 lines (16 loc) • 15.9 kB
JavaScript
'use strict';
Object.defineProperty(exports, '__esModule', { value: true });
var react = require('react');
var LexicalComposerContext = require('@lexical/react/LexicalComposerContext');
var lexical = require('lexical');
var b = require('axios');
var jsxRuntime = require('react/jsx-runtime');
function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
var b__default = /*#__PURE__*/_interopDefault(b);
var w=[{name:"shorten",value:["shorten"]},{name:"summarize",value:["summarize"]},{name:"simplify",value:["simplify"]},{name:"Splelling & Grammar",value:["spelling","grammar"]}],D=async(n,o,e)=>{if(!n||!o)return;let t=await b__default.default("https://lexical-ai-assistant.akashorasad2000.workers.dev/",{method:"POST",headers:{SECRET_KEY:n,"Content-Type":"application/json"},data:JSON.stringify({text:o,options:e})});return t.status===200?t.data:o};function d(n){let{parentRef:o,closeDropDown:e,onDropDownClick:t}=n,r=react.useRef(null);return react.useEffect(()=>{let a=r==null?void 0:r.current,s=o==null?void 0:o.current;if(a&&s){let p=u=>{let c=u.target;a.contains(c)||s!=null&&s.contains(c)||e();};return document.addEventListener("click",p),()=>{document.removeEventListener("click",p);}}},[r,o]),jsxRuntime.jsx("div",{className:"dropdown",ref:r,children:w.map((a,s)=>jsxRuntime.jsx("button",{className:"item",onClick:()=>{t(a.value);},children:jsxRuntime.jsx("span",{className:"text",children:a.name})},s))})}function m(n,{insertAt:o}={}){if(!n||typeof document>"u")return;let e=document.head||document.getElementsByTagName("head")[0],t=document.createElement("style");t.type="text/css",o==="top"&&e.firstChild?e.insertBefore(t,e.firstChild):e.appendChild(t),t.styleSheet?t.styleSheet.cssText=n:t.appendChild(document.createTextNode(n));}m(`.dropdown{z-index:5;display:block;position:absolute;box-shadow:0 12px 28px #0003,0 2px 4px #0000001a,inset 0 0 0 1px #ffffff80;border-radius:8px;min-width:100px;min-height:40px;background-color:#fff}.dropdown .item{margin:0 8px;padding:8px;color:#050505;cursor:pointer;line-height:16px;font-size:15px;display:flex;align-content:center;flex-direction:row;flex-shrink:0;justify-content:space-between;background-color:#fff;border-radius:8px;border:0;min-width:268px}.dropdown .item:first-child{margin-top:8px}.dropdown .item:last-child{margin-bottom:8px}.dropdown .item:hover{background-color:#eee}.dropdown .item .text{display:flex;line-height:20px;flex-grow:1;width:200px}.button{padding:2px 4px;cursor:pointer;outline:none;background:transparent;border:none}
`);var X=lexical.createCommand(),Z=lexical.createCommand();function R({apiKey:n,parentRef:o}){let[e]=LexicalComposerContext.useLexicalComposerContext(),[t,r]=react.useState(!1),a=react.useCallback(()=>{let i="";return e.update(()=>{i=lexical.$getTextContent();}),i},[e]),s=react.useCallback(async i=>{let x=a(),h=await D(n,x,i);e.update(()=>{let g=lexical.$getSelection();g&&h&&g.insertText(h);});},[e]),p=()=>{r(i=>!i);};return jsxRuntime.jsxs("div",{children:[jsxRuntime.jsx("button",{className:"button",onClick:()=>{p();},children:jsxRuntime.jsx("span",{children:"AI"})}),t&&jsxRuntime.jsx(d,{parentRef:o,closeDropDown:()=>{r(!1);},setShowOptions:r,onDropDownClick:i=>{s(i),p();}})]})}
exports.COPY = X;
exports.PASTE = Z;
exports.default = R;
//# sourceMappingURL=out.js.map
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/index.tsx","../src/util.ts","../src/dropdown.tsx","#style-inject:#style-inject","../src/index.css"],"names":["useCallback","useState","useLexicalComposerContext","createCommand","$getTextContent","$getSelection","axios","options","handleAIRequest","apiKey","text","response","useEffect","useRef","jsx","Dropdown","props","parentRef","closeDropDown","onDropDownClick","dropDownRef","dropDown","parentWrapper","handle","event","target","option","idx","styleInject","css","insertAt","head","style","jsxs","COPY","PASTE","AIPlugin","editor","showOptions","setShowOptions","getTextContent","handleAIAssistant","assitantResponse","selection","toggleDropDown","prevState"],"mappings":"AAAA,OAAgB,eAAAA,EAAa,YAAAC,MAAgB,QAC7C,OAAS,6BAAAC,MAAiC,wCAC1C,OAEE,iBAAAC,EACA,mBAAAC,EACA,iBAAAC,MACK,UCPP,OAAOC,MAAW,QAEX,IAAMC,EAAU,CACrB,CACE,KAAM,UACN,MAAO,CAAC,SAAS,CACnB,EACA,CACE,KAAM,YACN,MAAO,CAAC,WAAW,CACrB,EACA,CACE,KAAM,WACN,MAAO,CAAC,UAAU,CACpB,EACA,CACE,KAAM,sBACN,MAAO,CAAC,WAAY,SAAS,CAC/B,CACF,EAEaC,EAAkB,MAC7BC,EACAC,EACAH,IACG,CACH,GAAI,CAACE,GAAU,CAACC,EAAM,OAEtB,IAAMC,EAAW,MAAML,EACrB,4DACA,CACE,OAAQ,OACR,QAAS,CACP,WAAYG,EACZ,eAAgB,kBAClB,EACA,KAAM,KAAK,UAAU,CACnB,KAAAC,EACA,QAAAH,CACF,CAAC,CACH,CACF,EAEA,OAAII,EAAS,SAAW,IACfA,EAAS,KAGXD,CACT,EChDA,OAA0C,aAAAE,EAAW,UAAAC,MAAc,QA6CzD,cAAAC,MAAA,oBApCK,SAARC,EAA0BC,EAAc,CAC7C,GAAM,CAAE,UAAAC,EAAW,cAAAC,EAAe,gBAAAC,CAAgB,EAAIH,EAChDI,EAAcP,EAAuB,IAAI,EAE/C,OAAAD,EAAU,IAAM,CACd,IAAMS,EAAWD,GAAA,YAAAA,EAAa,QACxBE,EAAgBL,GAAA,YAAAA,EAAW,QAEjC,GAAII,GAAYC,EAAe,CAC7B,IAAMC,EAAUC,GAAsB,CACpC,IAAMC,EAASD,EAAM,OAEjBH,EAAS,SAASI,CAAM,GAAKH,GAAA,MAAAA,EAAe,SAASG,IAIzDP,EAAc,CAChB,EAEA,gBAAS,iBAAiB,QAASK,CAAM,EAElC,IAAM,CACX,SAAS,oBAAoB,QAASA,CAAM,CAC9C,CACF,CACF,EAAG,CAACH,EAAaH,CAAS,CAAC,EAEzBH,EAAC,OAAI,UAAU,WAAW,IAAKM,EAC5B,SAAAb,EAAQ,IAAI,CAACmB,EAAQC,IACpBb,EAAC,UAEC,UAAU,OACV,QAAS,IAAM,CACbK,EAAgBO,EAAO,KAAK,CAC9B,EAEA,SAAAZ,EAAC,QAAK,UAAU,OAAQ,SAAAY,EAAO,KAAK,GAN/BC,CAOP,CACD,EACH,CAEJ,CCjDyB,SAARC,EAA6BC,EAAK,CAAE,SAAAC,CAAS,EAAI,CAAC,EAAG,CAC1D,GAAI,CAACD,GAAO,OAAO,SAAa,IAAa,OAE7C,IAAME,EAAO,SAAS,MAAQ,SAAS,qBAAqB,MAAM,EAAE,CAAC,EAC/DC,EAAQ,SAAS,cAAc,OAAO,EAC5CA,EAAM,KAAO,WAETF,IAAa,OACXC,EAAK,WACPA,EAAK,aAAaC,EAAOD,EAAK,UAAU,EAK1CA,EAAK,YAAYC,CAAK,EAGpBA,EAAM,WACRA,EAAM,WAAW,QAAUH,EAE3BG,EAAM,YAAY,SAAS,eAAeH,CAAG,CAAC,CAElD,CCvB8BD,EAAY;AAAA,CAA0vB,EJiE1yB,OAEI,OAAAd,EAFJ,QAAAmB,MAAA,oBArDG,IAAMC,EAA+B/B,EAAc,EAC7CgC,EAAgChC,EAAc,EAO5C,SAARiC,EAA0B,CAAE,OAAA3B,EAAQ,UAAAQ,CAAU,EAAU,CAC7D,GAAM,CAACoB,CAAM,EAAInC,EAA0B,EACrC,CAACoC,EAAaC,CAAc,EAAItC,EAAS,EAAK,EAE9CuC,EAAiBxC,EAAY,IAAM,CACvC,IAAIU,EAAO,GACX,OAAA2B,EAAO,OAAO,IAAM,CAElB3B,EADoBN,EAAgB,CAEtC,CAAC,EACMM,CACT,EAAG,CAAC2B,CAAM,CAAC,EAELI,EAAoBzC,EACxB,MAAOO,GAA2B,CAChC,IAAMG,EAAO8B,EAAe,EACtBE,EAAmB,MAAMlC,EAAgBC,EAAQC,EAAMH,CAAO,EACpE8B,EAAO,OAAO,IAAM,CAClB,IAAMM,EAAYtC,EAAc,EAC5BsC,GAAaD,GACfC,EAAU,WAAWD,CAAgB,CAEzC,CAAC,CACH,EACA,CAACL,CAAM,CACT,EAEMO,EAAiB,IAAM,CAC3BL,EAAgBM,GAAc,CAACA,CAAS,CAC1C,EAeA,OACEZ,EAAC,OACC,UAAAnB,EAAC,UAAO,UAAU,SAAS,QAfV,IAAM,CACzB8B,EAAe,CACjB,EAcM,SAAA9B,EAAC,QAAK,cAAE,EACV,EACCwB,GACCxB,EAACC,EAAA,CACC,UAAWE,EACX,cAjBc,IAAM,CAC1BsB,EAAe,EAAK,CACtB,EAgBQ,eAAgBA,EAChB,gBAfgBhC,GAA2B,CACjDkC,EAAkBlC,CAAO,EACzBqC,EAAe,CACjB,EAaM,GAEJ,CAEJ","sourcesContent":["import React, { useCallback, useState } from \"react\";\nimport { useLexicalComposerContext } from \"@lexical/react/LexicalComposerContext\";\nimport {\n  LexicalCommand,\n  createCommand,\n  $getTextContent,\n  $getSelection,\n} from \"lexical\";\nimport { handleAIRequest } from \"./util\";\nimport Dropdown from \"./dropdown\";\nimport \"./index.css\";\n\nexport const COPY: LexicalCommand<string> = createCommand();\nexport const PASTE: LexicalCommand<string> = createCommand();\n\ntype Props = {\n  apiKey: string;\n  parentRef: React.RefObject<HTMLDivElement>;\n};\n\nexport default function AIPlugin({ apiKey, parentRef }: Props) {\n  const [editor] = useLexicalComposerContext();\n  const [showOptions, setShowOptions] = useState(false);\n\n  const getTextContent = useCallback(() => {\n    let text = \"\";\n    editor.update(() => {\n      const textContent = $getTextContent();\n      text = textContent;\n    });\n    return text;\n  }, [editor]);\n\n  const handleAIAssistant = useCallback(\n    async (options: Array<string>) => {\n      const text = getTextContent();\n      const assitantResponse = await handleAIRequest(apiKey, text, options);\n      editor.update(() => {\n        const selection = $getSelection();\n        if (selection && assitantResponse) {\n          selection.insertText(assitantResponse);\n        }\n      });\n    },\n    [editor]\n  );\n\n  const toggleDropDown = () => {\n    setShowOptions((prevState) => !prevState);\n  };\n\n  const openDropDown = () => {\n    toggleDropDown();\n  };\n\n  const closeDropDown = () => {\n    setShowOptions(false);\n  };\n\n  const handleDropDown = (options: Array<string>) => {\n    handleAIAssistant(options);\n    toggleDropDown();\n  };\n\n  return (\n    <div>\n      <button className=\"button\" onClick={openDropDown}>\n        <span>AI</span>\n      </button>\n      {showOptions && (\n        <Dropdown\n          parentRef={parentRef}\n          closeDropDown={closeDropDown}\n          setShowOptions={setShowOptions}\n          onDropDownClick={handleDropDown}\n        />\n      )}\n    </div>\n  );\n}\n","import axios from \"axios\";\n\nexport const options = [\n  {\n    name: \"shorten\",\n    value: [\"shorten\"],\n  },\n  {\n    name: \"summarize\",\n    value: [\"summarize\"],\n  },\n  {\n    name: \"simplify\",\n    value: [\"simplify\"],\n  },\n  {\n    name: \"Splelling & Grammar\",\n    value: [\"spelling\", \"grammar\"],\n  },\n];\n\nexport const handleAIRequest = async (\n  apiKey: string,\n  text: string,\n  options: Array<string>\n) => {\n  if (!apiKey || !text) return;\n\n  const response = await axios(\n    \"https://lexical-ai-assistant.akashorasad2000.workers.dev/\",\n    {\n      method: \"POST\",\n      headers: {\n        SECRET_KEY: apiKey,\n        \"Content-Type\": \"application/json\",\n      },\n      data: JSON.stringify({\n        text,\n        options,\n      }),\n    }\n  );\n\n  if (response.status === 200) {\n    return response.data;\n  }\n\n  return text;\n};\n","import React, { Dispatch, SetStateAction, useEffect, useRef } from \"react\";\nimport { options } from \"./util\";\ninterface Props {\n  parentRef: React.RefObject<HTMLDivElement>;\n  closeDropDown: () => void;\n  setShowOptions: Dispatch<SetStateAction<boolean>>;\n  onDropDownClick: (options: Array<string>) => void;\n}\n\nexport default function Dropdown(props: Props) {\n  const { parentRef, closeDropDown, onDropDownClick } = props;\n  const dropDownRef = useRef<HTMLDivElement>(null);\n\n  useEffect(() => {\n    const dropDown = dropDownRef?.current;\n    const parentWrapper = parentRef?.current;\n\n    if (dropDown && parentWrapper) {\n      const handle = (event: MouseEvent) => {\n        const target = event.target as Node;\n\n        if (dropDown.contains(target) || parentWrapper?.contains(target)) {\n          return;\n        }\n\n        closeDropDown();\n      };\n\n      document.addEventListener(\"click\", handle);\n\n      return () => {\n        document.removeEventListener(\"click\", handle);\n      };\n    }\n  }, [dropDownRef, parentRef]);\n  return (\n    <div className=\"dropdown\" ref={dropDownRef}>\n      {options.map((option, idx) => (\n        <button\n          key={idx}\n          className=\"item\"\n          onClick={() => {\n            onDropDownClick(option.value);\n          }}\n        >\n          <span className=\"text\">{option.name}</span>\n        </button>\n      ))}\n    </div>\n  );\n}\n","\n          export default function styleInject(css, { insertAt } = {}) {\n            if (!css || typeof document === 'undefined') return\n          \n            const head = document.head || document.getElementsByTagName('head')[0]\n            const style = document.createElement('style')\n            style.type = 'text/css'\n          \n            if (insertAt === 'top') {\n              if (head.firstChild) {\n                head.insertBefore(style, head.firstChild)\n              } else {\n                head.appendChild(style)\n              }\n            } else {\n              head.appendChild(style)\n            }\n          \n            if (style.styleSheet) {\n              style.styleSheet.cssText = css\n            } else {\n              style.appendChild(document.createTextNode(css))\n            }\n          }\n          ","import styleInject from '#style-inject';styleInject(\".dropdown{z-index:5;display:block;position:absolute;box-shadow:0 12px 28px #0003,0 2px 4px #0000001a,inset 0 0 0 1px #ffffff80;border-radius:8px;min-width:100px;min-height:40px;background-color:#fff}.dropdown .item{margin:0 8px;padding:8px;color:#050505;cursor:pointer;line-height:16px;font-size:15px;display:flex;align-content:center;flex-direction:row;flex-shrink:0;justify-content:space-between;background-color:#fff;border-radius:8px;border:0;min-width:268px}.dropdown .item:first-child{margin-top:8px}.dropdown .item:last-child{margin-bottom:8px}.dropdown .item:hover{background-color:#eee}.dropdown .item .text{display:flex;line-height:20px;flex-grow:1;width:200px}.button{padding:2px 4px;cursor:pointer;outline:none;background:transparent;border:none}\\n\")"]}