reneco-advanced-input-module 0.0.1
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/.editorconfig +15 -0
- package/.prettierrc.json +13 -0
- package/LICENSE +21 -0
- package/api-key-inject.js +46 -0
- package/dist/cjs/app-globals-V2Kpy_OQ.js +8 -0
- package/dist/cjs/app-globals-V2Kpy_OQ.js.map +1 -0
- package/dist/cjs/file-uploader.voice-input-module.entry.cjs.js.map +1 -0
- package/dist/cjs/file-uploader_2.cjs.entry.js +1319 -0
- package/dist/cjs/file-uploader_2.cjs.entry.js.map +1 -0
- package/dist/cjs/index-BTSzTkSZ.js +1494 -0
- package/dist/cjs/index-BTSzTkSZ.js.map +1 -0
- package/dist/cjs/index.cjs.js +5 -0
- package/dist/cjs/index.cjs.js.map +1 -0
- package/dist/cjs/loader.cjs.js +16 -0
- package/dist/cjs/loader.cjs.js.map +1 -0
- package/dist/cjs/voice-input-module.cjs.js +28 -0
- package/dist/cjs/voice-input-module.cjs.js.map +1 -0
- package/dist/collection/collection-manifest.json +13 -0
- package/dist/collection/components/file-uploader/file-uploader.css +26 -0
- package/dist/collection/components/file-uploader/file-uploader.js +130 -0
- package/dist/collection/components/file-uploader/file-uploader.js.map +1 -0
- package/dist/collection/components/voice-input-module/voice-input-module.css +251 -0
- package/dist/collection/components/voice-input-module/voice-input-module.js +875 -0
- package/dist/collection/components/voice-input-module/voice-input-module.js.map +1 -0
- package/dist/collection/index.js +12 -0
- package/dist/collection/index.js.map +1 -0
- package/dist/collection/services/audio-recorder.service.js +66 -0
- package/dist/collection/services/audio-recorder.service.js.map +1 -0
- package/dist/collection/services/llm.service.js +193 -0
- package/dist/collection/services/llm.service.js.map +1 -0
- package/dist/collection/services/speech-to-text.service.js +62 -0
- package/dist/collection/services/speech-to-text.service.js.map +1 -0
- package/dist/collection/types/form-schema.types.js +2 -0
- package/dist/collection/types/form-schema.types.js.map +1 -0
- package/dist/collection/types/service-providers.types.js +2 -0
- package/dist/collection/types/service-providers.types.js.map +1 -0
- package/dist/collection/utils/schema-converter.js +422 -0
- package/dist/collection/utils/schema-converter.js.map +1 -0
- package/dist/components/file-uploader.d.ts +11 -0
- package/dist/components/file-uploader.js +9 -0
- package/dist/components/file-uploader.js.map +1 -0
- package/dist/components/file-uploader2.js +98 -0
- package/dist/components/file-uploader2.js.map +1 -0
- package/dist/components/index.d.ts +33 -0
- package/dist/components/index.js +4 -0
- package/dist/components/index.js.map +1 -0
- package/dist/components/voice-input-module.d.ts +11 -0
- package/dist/components/voice-input-module.js +1292 -0
- package/dist/components/voice-input-module.js.map +1 -0
- package/dist/esm/app-globals-DQuL1Twl.js +6 -0
- package/dist/esm/app-globals-DQuL1Twl.js.map +1 -0
- package/dist/esm/file-uploader.voice-input-module.entry.js.map +1 -0
- package/dist/esm/file-uploader_2.entry.js +1316 -0
- package/dist/esm/file-uploader_2.entry.js.map +1 -0
- package/dist/esm/index-jmc2yzBp.js +1487 -0
- package/dist/esm/index-jmc2yzBp.js.map +1 -0
- package/dist/esm/index.js +4 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/loader.js +14 -0
- package/dist/esm/loader.js.map +1 -0
- package/dist/esm/voice-input-module.js +24 -0
- package/dist/esm/voice-input-module.js.map +1 -0
- package/dist/index.cjs.js +1 -0
- package/dist/index.js +1 -0
- package/dist/types/components/file-uploader/file-uploader.d.ts +8 -0
- package/dist/types/components/voice-input-module/voice-input-module.d.ts +55 -0
- package/dist/types/components.d.ts +158 -0
- package/dist/types/index.d.ts +9 -0
- package/dist/types/services/audio-recorder.service.d.ts +9 -0
- package/dist/types/services/llm.service.d.ts +15 -0
- package/dist/types/services/speech-to-text.service.d.ts +11 -0
- package/dist/types/stencil-public-runtime.d.ts +1709 -0
- package/dist/types/types/form-schema.types.d.ts +70 -0
- package/dist/types/types/service-providers.types.d.ts +20 -0
- package/dist/types/utils/schema-converter.d.ts +22 -0
- package/dist/voice-input-module/file-uploader.voice-input-module.entry.esm.js.map +1 -0
- package/dist/voice-input-module/index.esm.js +2 -0
- package/dist/voice-input-module/index.esm.js.map +1 -0
- package/dist/voice-input-module/loader.esm.js.map +1 -0
- package/dist/voice-input-module/p-7b4f33ba.entry.js +2 -0
- package/dist/voice-input-module/p-7b4f33ba.entry.js.map +1 -0
- package/dist/voice-input-module/p-DQuL1Twl.js +2 -0
- package/dist/voice-input-module/p-DQuL1Twl.js.map +1 -0
- package/dist/voice-input-module/p-jmc2yzBp.js +3 -0
- package/dist/voice-input-module/p-jmc2yzBp.js.map +1 -0
- package/dist/voice-input-module/voice-input-module.esm.js +2 -0
- package/dist/voice-input-module/voice-input-module.esm.js.map +1 -0
- package/env-config.js +4 -0
- package/inject-env.js +20 -0
- package/package.json +37 -0
- package/readme.md +111 -0
- package/src/components/file-uploader/file-uploader.css +26 -0
- package/src/components/file-uploader/file-uploader.tsx +100 -0
- package/src/components/file-uploader/readme.md +31 -0
- package/src/components/voice-input-module/readme.md +114 -0
- package/src/components/voice-input-module/voice-input-module.css +251 -0
- package/src/components/voice-input-module/voice-input-module.tsx +731 -0
- package/src/components.d.ts +158 -0
- package/src/index.html +663 -0
- package/src/index.ts +12 -0
- package/src/services/audio-recorder.service.ts +74 -0
- package/src/services/llm.service.ts +221 -0
- package/src/services/speech-to-text.service.ts +72 -0
- package/src/types/form-schema.types.ts +78 -0
- package/src/types/service-providers.types.ts +22 -0
- package/src/utils/schema-converter.ts +494 -0
- package/stencil.config.ts +24 -0
- package/tsconfig.json +30 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"voice-input-module.js","sourceRoot":"","sources":["../../../src/components/voice-input-module/voice-input-module.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAgB,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAG9F,OAAO,EAAE,oBAAoB,EAAE,MAAM,uCAAuC,CAAC;AAC7E,OAAO,EAAE,0BAA0B,EAAE,MAAM,uCAAuC,CAAC;AACnF,OAAO,EAAE,iBAAiB,EAAE,MAAM,4BAA4B,CAAC;AAC/D,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAO/D,MAAM,OAAO,iBAAiB;IA+B5B;QA9BQ,aAAQ,GAAW,IAAI,CAAC;QACxB,kBAAa,GAAW,IAAI,CAAC;QAE7B,YAAO,GAAwC,SAAS,CAAC;QACzD,0BAAqB,GAAW,kBAAkB,CAAC;QACnD,aAAQ,GAAc,IAAI,CAAC;QAC3B,eAAU,GAAyB,SAAS,CAAC;QAE7C,UAAK,GAAY,KAAK,CAAC;QACvB,eAAU,GAAY,KAAK,CAAC;QAC5B,kBAAa,GAAY,KAAK,CAAC;QAK9B,gBAAW,GAAY,KAAK,CAAC;QAC7B,iBAAY,GAAY,KAAK,CAAC;QAE9B,aAAQ,GAAY,KAAK,CAAC;QAC1B,kBAAa,GAAW,EAAE,CAAC;QAC3B,eAAU,GAA0B,IAAI,CAAC;QACzC,cAAS,GAAQ,EAAE,CAAC;QACpB,mBAAc,GAAY,IAAI,CAAC,CAAC,iCAAiC;QASxE,IAAI,CAAC,aAAa,GAAG,IAAI,oBAAoB,EAAE,CAAC;KACjD;IAED,iBAAiB;QACf,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAID,kBAAkB;QAChB,IAAI,CAAC;YACH,IAAI,CAAC,IAAI,CAAC,OAAO,EAAC,CAAC;gBACjB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;gBACrB,MAAM,YAAY,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,qCAAqC,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC,6CAA6C,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC;gBACnK,IAAI,CAAC,aAAa,GAAG,YAAY,CAAC;gBAClC,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAC9D,CAAC;iBACG,CAAC;gBACH,oBAAoB;gBACpB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC;gBAEtD,8BAA8B;gBAC9B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,CAAC;gBAE3D,6CAA6C;gBAC7C,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;oBAChB,IAAI,CAAC,YAAY,mCACZ,IAAI,CAAC,YAAY,KACpB,YAAY,kCAAO,IAAI,CAAC,YAAY,CAAC,YAAY,KAAE,MAAM,EAAE,IAAI,CAAC,MAAM,KACtE,GAAG,kCAAO,IAAI,CAAC,YAAY,CAAC,GAAG,KAAE,MAAM,EAAE,IAAI,CAAC,MAAM,MACrD,CAAC;gBACJ,CAAC;gBAED,sBAAsB;gBACtB,IAAI,CAAC,mBAAmB,GAAG,0BAA0B,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBAChF,IAAI,CAAC,UAAU,GAAG,iBAAiB,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBAE9D,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE;oBAClC,MAAM,EAAE,IAAI,CAAC,YAAY;oBACzB,MAAM,EAAE,IAAI,CAAC,YAAY;iBAC1B,CAAC,CAAC;gBAEH,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;gBACtB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC;YACzG,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,yBAAyB,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,4BAA4B,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YACtI,IAAI,CAAC,eAAe,CAAC,sBAAsB,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QACzE,CAAC;IACH,CAAC;IAEO,eAAe,CAAC,MAAc,EAAE,IAAS;QAC/C,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,SAAS,mCACT,IAAI,CAAC,SAAS,KACjB,CAAC,MAAM,CAAC,EAAE;oBACR,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;oBACnC,IAAI;iBACL,GACF,CAAC;QACJ,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,iBAAiB;QAC7B,IAAI,IAAI,CAAC,YAAY;YAAE,OAAO;QAE9B,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,MAAM,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACvC,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAC9B,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,cAAc;QAC1B,IAAI,CAAC;YACH,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,uBAAuB,CAAC,CAAC,CAAC,oBAAoB,CAAC,CAAC;YAC9F,IAAI,CAAC,eAAe,CAAC,yBAAyB,EAAE,EAAE,CAAC,CAAC;YAEpD,MAAM,IAAI,CAAC,aAAa,CAAC,cAAc,EAAE,CAAC;YAE1C,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;YACxB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,4BAA4B,CAAC,CAAC,CAAC,yCAAyC,CAAC,CAAC;YACxH,IAAI,CAAC,eAAe,CAAC,mBAAmB,EAAE,EAAE,CAAC,CAAC;YAE9C,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;gBAC9B,WAAW,EAAE,IAAI;gBACjB,KAAK,EAAE,WAAW;aACnB,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,qBAAqB,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,+BAA+B,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YACrI,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YAElE,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;gBACnB,OAAO,EAAE,KAAK;gBACd,KAAK,EAAE,KAAK,CAAC,OAAO;aACrB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,eAAe,CAAC,QAAgB;QAC5C,4CAA4C;QAC5C,IAAI,CAAC;YACH,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,oBAAoB,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC;YAE/F,sBAAsB;YACtB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,+BAA+B,CAAC,CAAC;YAC1G,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC9D,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,QAAQ,EAAE,aAAa,CAAC,CAAC;YAErF,sBAAsB;YACtB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;YACvD,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE;gBAClC,YAAY;gBACZ,aAAa,EAAE,IAAI,CAAC,UAAU;aAC/B,CAAC,CAAC;YAEH,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC;YAEpC,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC;YAC1F,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YAEtB,qBAAqB;YACrB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;gBACnB,OAAO,EAAE,IAAI;gBACb,IAAI,EAAE,IAAI,CAAC,UAAU;gBACrB,QAAQ,EAAE,QAAQ;aACnB,CAAC,CAAC;QAEL,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,sBAAsB,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,0BAA0B,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YAEjI,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;gBACnB,OAAO,EAAE,KAAK;gBACd,KAAK,EAAE,KAAK,CAAC,OAAO;gBACpB,QAAQ,EAAE,QAAQ;aACnB,CAAC,CAAC;QACL,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC5B,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,uBAAuB;QACnC,IAAI,CAAC;YACH,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YACzB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,2BAA2B,CAAC,CAAC;YACnG,IAAI,CAAC,eAAe,CAAC,gBAAgB,EAAE,EAAE,CAAC,CAAC;YAE3C,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;gBAC9B,WAAW,EAAE,KAAK;gBAClB,KAAK,EAAE,YAAY;aACpB,CAAC,CAAC;YAEH,oCAAoC;YACpC,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,aAAa,EAAE,CAAC;YAC3D,IAAI,CAAC,eAAe,CAAC,gBAAgB,EAAE;gBACrC,IAAI,EAAE,SAAS,CAAC,IAAI;gBACpB,IAAI,EAAE,SAAS,CAAC,IAAI;aACrB,CAAC,CAAC;YAEH,mBAAmB;YACnB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,4BAA4B,CAAC,CAAC;YACvG,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,UAAU,CAAC,SAAS,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC1F,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;YACnC,IAAI,CAAC,eAAe,CAAC,wBAAwB,EAAE,EAAE,aAAa,EAAE,CAAC,CAAC;YAElE,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,EAAE,CAAC;gBAC1B,MAAM,IAAI,KAAK,CAAC,qCAAqC,CAAC,CAAC;YACzD,CAAC;YAED,sBAAsB;YACtB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,+BAA+B,CAAC,CAAC;YAC1G,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC9D,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,yBAAyB,CAAC,aAAa,EAAE,aAAa,CAAC,CAAC;YAEnG,sBAAsB;YACtB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;YACvD,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE;gBAClC,YAAY;gBACZ,aAAa,EAAE,IAAI,CAAC,UAAU;aAC/B,CAAC,CAAC;YAEH,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC;YAEpC,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC;YAC1F,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YAEtB,qBAAqB;YACrB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;gBACnB,OAAO,EAAE,IAAI;gBACb,IAAI,EAAE,IAAI,CAAC,UAAU;gBACrB,aAAa,EAAE,aAAa;aAC7B,CAAC,CAAC;QAEL,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,sBAAsB,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,0BAA0B,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YACjI,IAAI,CAAC,eAAe,CAAC,kBAAkB,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YAEnE,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;gBACnB,OAAO,EAAE,KAAK;gBACd,KAAK,EAAE,KAAK,CAAC,OAAO;gBACpB,aAAa,EAAE,IAAI,CAAC,aAAa;aAClC,CAAC,CAAC;QACL,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;YAC1B,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;gBAC9B,WAAW,EAAE,KAAK;gBAClB,KAAK,EAAE,MAAM;aACd,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,iBAAiB,CAAC,UAAe;QACvC,gDAAgD;QAChD,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;QACpE,QAAO,IAAI,CAAC,OAAO,EAAC,CAAC;YACnB,KAAK,IAAI;gBACP,IAAI,UAAU,aAAV,UAAU,uBAAV,UAAU,CAAE,MAAM,EAAE,CAAC;oBACvB,oDAAoD;oBACpD,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAU,EAAE,EAAE;wBACvC,MAAM,aAAa,GAAG,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,KAAU,EAAE,EAAE,CAC/D,KAAK,CAAC,WAAW,KAAK,KAAK,CAAC,IAAI,CACjC,CAAC;wBAEF,IAAI,aAAa,IAAI,KAAK,CAAC,KAAK,KAAK,SAAS,IAAI,KAAK,CAAC,KAAK,KAAK,IAAI,IAAI,KAAK,CAAC,KAAK,KAAK,EAAE,EAAE,CAAC;4BAC7F,IAAI,CAAC,aAAa,CAAC,QAAQ;gCAAE,aAAa,CAAC,QAAQ,GAAG,EAAE,CAAC;4BACzD,aAAa,CAAC,QAAQ,CAAC,aAAa,GAAG,KAAK,CAAC,KAAK,CAAC;wBACrD,CAAC;oBACH,CAAC,CAAC,CAAC;gBACL,CAAC;gBAED,MAAM;YACR,KAAK,YAAY;gBACf,oEAAoE;gBACpE,IAAI,UAAU,aAAV,UAAU,uBAAV,UAAU,CAAE,MAAM,EAAE,CAAC;oBACvB,oDAAoD;oBACpD,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAU,EAAE,EAAE;wBACvC,IAAI,aAAa,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,KAAU,EAAE,EAAE,CAC7D,KAAK,CAAC,KAAK,KAAK,KAAK,CAAC,IAAI,CAC3B,CAAC;wBACF,IAAI,CAAC,aAAa;4BACf,aAAa,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,KAAU,EAAE,EAAE,CAC1D,KAAK,CAAC,KAAK,KAAK,KAAK,CAAC,IAAI,CAC3B,CAAC;wBAEJ,IAAI,aAAa,IAAI,KAAK,CAAC,KAAK,KAAK,SAAS,IAAI,KAAK,CAAC,KAAK,KAAK,IAAI,IAAI,KAAK,CAAC,KAAK,KAAK,EAAE,EAAE,CAAC;4BAC7F,aAAa,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;wBACrD,CAAC;oBACH,CAAC,CAAC,CAAC;gBACL,CAAC;gBACD,MAAM;YACR,KAAK,SAAS;gBACZ,qDAAqD;gBACrD,MAAM;YACR,KAAK,OAAO,CAAC;YACb;gBACE,MAAM,IAAI,GAAmB,EAAE,CAAC;gBAEhC,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,EAAE,KAAK,CAAgB,EAAE,EAAE;oBAC5E,IAAI,KAAK,CAAC,OAAO,KAAK,SAAS,IAAI,KAAK,CAAC,OAAO,KAAK,IAAI,IAAI,KAAK,CAAC,OAAO,KAAK,EAAE,EAAE,CAAC;wBAClF,aAAa,CAAC,OAAO,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;oBACzC,CAAC;oBACD,IAAI,KAAK,CAAC,KAAK,KAAK,SAAS,IAAI,KAAK,CAAC,KAAK,KAAK,IAAI,IAAI,KAAK,CAAC,KAAK,KAAK,EAAE,EAAE,CAAC;wBAC5E,KAAK,MAAM,GAAG,IAAI,aAAa,CAAC,MAAM,EAAE,CAAC;4BACvC,MAAM,WAAW,GAAG,aAAa,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;4BAC9C,IAAI,WAAW,CAAC,KAAK,KAAK,KAAK,CAAC,IAAI,EAAE,CAAC;gCACrC,WAAW,CAAC,KAAK,GAAG,KAAK,CAAC,KAAK,IAAI,KAAK,CAAC,OAAO,CAAC;gCACjD,WAAW,CAAC,OAAO,GAAG,KAAK,CAAC,KAAK,IAAI,KAAK,CAAC,OAAO,CAAC;gCACnD,MAAM,CAAC,qCAAqC;4BAC9C,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC,CAAC,CAAC;gBAEH,MAAM;QACV,CAAC;QAED,0DAA0D;QAC1D,OAAO,aAAa,CAAC;IACvB,CAAC;IAEO,eAAe,CAAC,MAAW;;QACjC,0CAA0C;QAC1C,QAAO,IAAI,CAAC,OAAO,EAAC,CAAC;YACnB,KAAK,IAAI;gBACP,MAAM,OAAO,GAAG,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;gBAC/B,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,KAAU,EAAE,EAAE;oBACrC,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,CAAC,KAAK,CAAC,IAAI;wBAAE,OAAO;oBAE9C,MAAM,SAAS,GAAQ;wBACrB,IAAI,EAAE,KAAK,CAAC,WAAW;wBACvB,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC;qBACpC,CAAC;oBAEF,+CAA+C;oBAC/C,IAAI,KAAK,CAAC,IAAI,KAAK,qBAAqB,IAAI,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;wBACxF,SAAS,CAAC,OAAO,GAAG,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,MAAW,EAAE,EAAE,CACrD,MAAM,CAAC,WAAW,IAAI,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,QAAQ,EAAE,CACxD,CAAC;oBACJ,CAAC;oBAED,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBACjC,CAAC,CAAC,CAAC;gBAEH,qEAAqE;gBACrE,OAAO,OAAO,CAAC;YACjB,KAAK,YAAY;gBACf,8CAA8C;gBAC9C,MAAM,iBAAiB,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;gBAE1F,IAAI,iBAAiB,EAAE,CAAC;oBACtB,MAAM,aAAa,GAAe;wBAChC,KAAK,EAAE,WAAW;wBAClB,WAAW,EAAE,kBAAkB;wBAC/B,MAAM,EAAE,EAAE;qBACX,CAAC;oBAEF,MAAM,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAgB,EAAE,EAAE;wBACxE,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC;wBAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,CAAQ,CAAC;wBACvD,MAAM,UAAU,GAAG,KAAK,CAAC,KAAK,IAAI,SAAS,CAAC;wBAC5C,MAAM,UAAU,GAAG,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC;wBAC3C,MAAM,UAAU,GAAG,EAAE,CAAC,CAAC,MAAM;wBAE7B,aAAa,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG;4BAChC,IAAI,EAAE,SAAS;4BACf,KAAK,EAAE,UAAU;4BACjB,OAAO,EAAE,KAAK,CAAC,OAAO;4BACtB,QAAQ,EAAE,KAAK,CAAC,QAAQ,KAAK,IAAI;4BACjC,OAAO,EAAE,EAAE;yBACZ,CAAC;oBACJ,CAAC,CAAC,CAAC;oBAEH,8EAA8E;oBAC9E,OAAO,aAAa,CAAC;gBACvB,CAAC;YACH,KAAK,SAAS;gBACZ,8CAA8C;gBAC9C,MAAM;YACR,KAAK,OAAO,CAAC;YACb;gBACE,uDAAuD;gBACvD,IAAI,MAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,MAAM,mCAAI,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,MAAM,EAAE,CAAC;oBACrC,MAAM,aAAa,GAAe;wBAChC,KAAK,EAAE,MAAM,CAAC,KAAK;wBACnB,WAAW,EAAE,MAAM,CAAC,WAAW;wBAC/B,MAAM,EAAE,EAAE;qBACX,CAAC;oBAEF,MAAM,WAAW,GAAG,MAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,MAAM,mCAAI,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,MAAM,CAAC;oBAErD,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,KAAK,CAAgB,EAAE,EAAE;wBACxE,aAAa,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG;4BAChC,IAAI,EAAE,KAAK,CAAC,IAAI;4BAChB,KAAK,EAAE,KAAK,CAAC,KAAK;4BAClB,OAAO,EAAE,KAAK,CAAC,OAAO;4BACtB,QAAQ,EAAE,KAAK,CAAC,OAAO,KAAK,KAAK;4BACjC,OAAO,EAAE,KAAK,CAAC,YAAY;4BAC3B,OAAO,EAAE,KAAK,CAAC,IAAI;4BACnB,GAAG,EAAE,KAAK,CAAC,aAAa;4BACxB,GAAG,EAAE,KAAK,CAAC,aAAa;yBACzB,CAAC;oBACJ,CAAC,CAAC,CAAC;oBAGH,8EAA8E;oBAC9E,OAAO,aAAa,CAAC;gBACvB,CAAC;gBACD,MAAM;QACV,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAEO,YAAY,CAAC,IAAY;QAC/B,MAAM,WAAW,GAAG;YAClB,mBAAmB,EAAE,QAAQ;YAC7B,cAAc,EAAE,QAAQ;YACxB,eAAe,EAAE,QAAQ;YACzB,qBAAqB,EAAE,UAAU;YACjC,cAAc,EAAE,QAAQ;YACxB,qBAAqB,EAAE,QAAQ;YAC/B,eAAe,EAAE,SAAS;YAC1B,sBAAsB,EAAE,QAAQ;YAChC,WAAW,EAAE,QAAQ;YACrB,UAAU,EAAE,QAAQ;YACpB,MAAM,EAAE,QAAQ;YAChB,UAAU,EAAE,QAAQ;YACpB,QAAQ,EAAE,QAAQ;YAClB,MAAM,EAAE,MAAM;YACd,UAAU,EAAE,UAAU;SACvB,CAAC;QAEF,OAAO,WAAW,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC;IACvC,CAAC;IAED,wCAAwC;IACvB,KAAK,CAAC,gBAAgB,CAAC,OAAe;QACrD,OAAO,eAAe,CAAC,gBAAgB,CAAC,OAAO,EAAE,IAAI,CAAC,qBAAqB,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC9F,CAAC;IAEgB,KAAK,CAAC,gBAAgB,CAAC,QAAsC;QAC5E,OAAO,eAAe,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;IACpD,CAAC;IAED,wCAAwC;IACvB,KAAK,CAAC,sBAAsB,CAAC,OAAe;QAC3D,OAAO,eAAe,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;IACzD,CAAC;IAEgB,KAAK,CAAC,sBAAsB,CAAC,QAAoB;QAChE,OAAO,eAAe,CAAC,sBAAsB,CAAC,QAAQ,CAAC,CAAC;IAC1D,CAAC;IAEO,kBAAkB;QACxB,IAAI,CAAC,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC;YAC5C,OAAO;QACT,OAAM,CACJ,qBACE,KAAK,EAAE,KAAK,EACZ,QAAQ,EAAE,CAAC,IAAI,EAAE,EAAE,GAAE,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAA,CAAC,GAEnC,CACjB,CAAC;IACJ,CAAC;IAEO,kBAAkB;QACxB,IAAI,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC;YACzD,OAAO;QAET,MAAM,WAAW,GAAG;YAClB,eAAe;YACf,IAAI,CAAC,WAAW,IAAI,WAAW;YAC/B,IAAI,CAAC,YAAY,IAAI,YAAY;SAClC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAE5B,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,QAAQ,CAAC;QAEtD,OAAO,CACL,cACE,KAAK,EAAE,WAAW,EAClB,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,EACvC,QAAQ,EAAE,UAAU,gBACR,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,iBAAiB,IAElE,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CACnB,WAAK,KAAK,EAAC,aAAa,EAAC,OAAO,EAAC,WAAW;YAC1C,cAAQ,EAAE,EAAC,IAAI,EAAC,EAAE,EAAC,IAAI,EAAC,CAAC,EAAC,GAAG;gBAC3B,eAAS,aAAa,EAAC,GAAG,EAAC,MAAM,EAAC,OAAO,EAAC,GAAG,EAAC,IAAI,EAAC,WAAW,EAAC,YAAY,GAAG;gBAC9E,eAAS,aAAa,EAAC,SAAS,EAAC,MAAM,EAAC,SAAS,EAAC,GAAG,EAAC,IAAI,EAAC,WAAW,EAAC,YAAY,GAAG,CAC/E,CACL,CACP,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CACrB,WAAK,KAAK,EAAC,aAAa,EAAC,OAAO,EAAC,WAAW;YAC1C,YAAM,CAAC,EAAC,GAAG,EAAC,CAAC,EAAC,GAAG,EAAC,KAAK,EAAC,IAAI,EAAC,MAAM,EAAC,IAAI,EAAC,EAAE,EAAC,GAAG,GAAG,CAC9C,CACP,CAAC,CAAC,CAAC,CACF,WAAK,KAAK,EAAC,aAAa,EAAC,OAAO,EAAC,WAAW;YAC1C,cAAQ,EAAE,EAAC,IAAI,EAAC,EAAE,EAAC,IAAI,EAAC,CAAC,EAAC,GAAG,GAAG,CAC5B,CACP,CACM,CACV,CAAC;IACJ,CAAC;IAEO,mBAAmB;QACzB,MAAM,WAAW,GAAG;YAClB,aAAa;YACb,IAAI,CAAC,QAAQ,IAAI,OAAO;YACxB,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,SAAS;SAC/C,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAE5B,OAAO,WAAK,KAAK,EAAE,WAAW,IAAG,IAAI,CAAC,aAAa,CAAO,CAAC;IAC7D,CAAC;IAEO,iBAAiB;QACvB,IAAI,CAAC,IAAI,CAAC,YAAY;YAAE,OAAO,IAAI,CAAC;QAEpC,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;QAC1D,MAAM,KAAK,GAAG,SAAS,CAAC,CAAC,CAAC,oCAAoC,CAAC,CAAC,CAAC,oBAAoB,CAAC;QAEtF,OAAO,CACL,WAAK,KAAK,EAAC,cAAc;YACvB,WAAK,KAAK,EAAC,oBAAoB,IAAE,KAAK,CAAO;YAC7C,YAAM,KAAK,EAAC,mBAAmB,IAC5B,IAAI,CAAC,gBAAgB,EAAE,CACnB,CACH,CACP,CAAC;IACJ,CAAC;IAEO,gBAAgB;QACtB,sDAAsD;QACtD,IAAI,CAAC,IAAI,CAAC,YAAY;YAAE,OAAO,IAAI,CAAC;QAEpC,QAAO,IAAI,CAAC,OAAO,EAAC,CAAC;YACnB,KAAK,IAAI;gBACP,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,KAAU,EAAE,EAAE;;oBACnD,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,CAAC,KAAK,CAAC,IAAI;wBAAE,OAAO,IAAI,CAAC;oBAEnD,MAAM,SAAS,GAAG,KAAK,CAAC,WAAW,CAAC;oBACpC,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;oBAChD,MAAM,UAAU,GAAG,CAAA,MAAA,KAAK,CAAC,QAAQ,0CAAE,KAAK,KAAI,KAAK,CAAC,WAAW,CAAC;oBAC9D,MAAM,UAAU,GAAG,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC;oBAC3C,MAAM,UAAU,GAAG,MAAA,KAAK,CAAC,QAAQ,0CAAE,aAAa,CAAC;oBAEjD,OAAO,CACL,WAAK,KAAK,EAAC,YAAY,EAAC,GAAG,EAAE,SAAS;wBACpC,aAAO,OAAO,EAAE,SAAS,EAAE,KAAK,EAAC,YAAY;4BAC1C,UAAU;4BACV,UAAU,IAAI,YAAM,KAAK,EAAC,UAAU,QAAS,CACxC;wBACP,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE;4BAC/B,IAAI,EAAE,SAAS;4BACf,KAAK,EAAE,UAAU;4BACjB,QAAQ,EAAE,UAAU;4BACpB,OAAO,EAAE,MAAA,KAAK,CAAC,QAAQ,0CAAE,GAAG,CAAC,CAAC,MAAW,EAAE,EAAE,CAC3C,MAAM,CAAC,WAAW,IAAI,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,QAAQ,EAAE,CACxD;yBACF,EAAE,UAAU,CAAC,CACV,CACP,CAAC;gBACJ,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YACrB,KAAK,YAAY;gBACf,cAAc;gBACd,2DAA2D;gBAC3D,MAAM,iBAAiB,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;gBAC1F,OAAO,MAAM,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAgB,EAAE,EAAE;;oBAE3E,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC;oBAC7B,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;oBAC3C,MAAM,UAAU,GAAG,KAAK,CAAC,KAAK,IAAI,SAAS,CAAC;oBAC5C,MAAM,UAAU,GAAG,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC;oBAC3C,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;oBAEnD,OAAO,CACL,WAAK,KAAK,EAAC,YAAY,EAAC,GAAG,EAAE,SAAS;wBACpC,aAAO,OAAO,EAAE,SAAS,EAAE,KAAK,EAAC,YAAY;4BACzC,UAAU;4BACX,UAAU,IAAI,YAAM,KAAK,EAAC,UAAU,QAAS,CACxC;wBACP,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,KAAK,EAAE,CAAC,MAAA,MAAA,IAAI,CAAC,UAAU,0CAAG,SAAS,CAAC,mCAAI,UAAU,CAAC,CAAC,CACjF,CACP,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,KAAK,SAAS;gBACZ,2DAA2D;gBAC3D,MAAM;YACR,KAAK,OAAO,CAAC;YACb;gBACE,OAAO,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,EAAE,KAAK,CAAgB,EAAE,EAAE;;oBAAC,OAAA,CACzF,WAAK,KAAK,EAAC,YAAY,EAAC,GAAG,EAAE,SAAS;wBACpC,aAAO,OAAO,EAAE,SAAS,EAAE,KAAK,EAAC,YAAY;4BAC1C,KAAK,CAAC,KAAK,IAAI,SAAS;4BACxB,KAAK,CAAC,QAAQ,IAAI,YAAM,KAAK,EAAC,UAAU,QAAS,CAC5C;wBACP,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,KAAK,EAAE,CAAC,MAAA,MAAA,IAAI,CAAC,UAAU,0CAAG,SAAS,CAAC,mCAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAClF,CACP,CAAA;iBAAA,CAAC,CAAC;QACP,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,eAAe,CAAC,SAAiB,EAAE,KAAU,EAAE,KAAU;;QAC/D,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;QAC3D,MAAM,WAAW,GAAG;YAClB,EAAE,EAAE,SAAS;YACb,IAAI,EAAE,SAAS;YACf,KAAK,EAAE,YAAY;YACnB,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,QAAQ,EAAE,UAAU;SACrB,CAAC;QAEF,QAAQ,KAAK,CAAC,IAAI,EAAE,CAAC;YACnB,KAAK,QAAQ;gBACX,IAAI,UAAU,EAAE,CAAC;oBACf,mEAAmE;oBACnE,OAAO,CACL,WAAK,KAAK,EAAC,iBAAiB;wBAC1B,WAAK,KAAK,EAAC,oBAAoB,yBAAyB;wBACxD,UAAI,KAAK,EAAC,qBAAqB,IAC5B,MAAA,KAAK,CAAC,OAAO,0CAAE,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAC5B,UAAI,KAAK,EAAC,eAAe,IAAE,MAAM,CAAM,CACxC,CAAC,CACC,CACD,CACP,CAAC;gBACJ,CAAC;gBACD,OAAO,CACL,cAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,SAAS,EAAE,KAAK,EAAC,YAAY,EAAC,QAAQ,EAAE,KAAK,CAAC,QAAQ;oBACjF,cAAQ,KAAK,EAAC,EAAE,mBAAsB,EACrC,MAAA,KAAK,CAAC,OAAO;uBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAC5B,cAAQ,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,KAAK,MAAM,IAC9C,MAAM,CACA,CACV,CAAC,CACK,CACV,CAAC;YAEJ,KAAK,SAAS;gBACZ,OAAO,CACL,6BACM,WAAW,IACf,IAAI,EAAC,UAAU,EACf,KAAK,EAAC,eAAe,EACrB,OAAO,EAAE,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,MAAM,IAC3C,CACH,CAAC;YAEJ,KAAK,QAAQ;gBACX,OAAO,CACL,6BACM,WAAW,IACf,IAAI,EAAC,QAAQ,EACb,GAAG,EAAE,KAAK,CAAC,GAAG,EACd,GAAG,EAAE,KAAK,CAAC,GAAG,EACd,IAAI,EAAC,KAAK,EACV,KAAK,EAAE,KAAK,IAAI,EAAE,IAClB,CACH,CAAC;YAEJ,KAAK,MAAM;gBACT,OAAO,CACL,6BACM,WAAW,IACf,IAAI,EAAC,MAAM,EACX,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,IACvE,CACH,CAAC;YAEJ,KAAK,UAAU;gBACb,OAAO,CACL,6BACM,WAAW,IACf,IAAI,EAAC,gBAAgB,EACrB,KAAK,EAAE,KAAK,IAAI,EAAE,IAClB,CACH,CAAC;YAEJ,SAAS,SAAS;gBAChB,OAAO,CACL,6BACM,WAAW,IACf,IAAI,EAAC,MAAM,EACX,OAAO,EAAE,KAAK,CAAC,OAAO,EACtB,WAAW,EAAE,KAAK,CAAC,WAAW,EAC9B,KAAK,EAAE,KAAK,IAAI,EAAE,IAClB,CACH,CAAC;QACN,CAAC;IACH,CAAC;IAEO,gBAAgB;QACtB,IAAI,CAAC,IAAI,CAAC,KAAK;YAAE,OAAO,IAAI,CAAC;QAE7B,OAAO,CACL,WAAK,KAAK,EAAC,aAAa;YACtB,WAAK,KAAK,EAAC,aAAa,yBAAyB;YACjD,WAAK,KAAK,EAAC,eAAe,IACvB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC,CACpC,CACF,CACP,CAAC;IACJ,CAAC;IAED,MAAM;QACJ,OAAO,CACL;YACE,4DAAK,KAAK,EAAE,0BAA0B,GAAG,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC;gBACrF,IAAI,CAAC,kBAAkB,EAAE;gBACzB,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC,CAAC,EAAE;gBACpD,IAAI,CAAC,kBAAkB,EAAE;gBACzB,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC,EAAE;gBAC/C,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC,CAAC,EAAE,CACtC,CACF,CACP,CAAC;IACJ,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CACF","sourcesContent":["import { Component, Prop, Event, EventEmitter, State, h, Watch, Method } from '@stencil/core';\r\nimport { FormSchema, VoiceFormRecorderResult, FilledFormData, FormSchemaFieldsOnly, FormSchemaFieldsOnlyExtended } from '../../types/form-schema.types';\r\nimport { ServiceProviderConfig } from '../../types/service-providers.types';\r\nimport { AudioRecorderService } from '../../services/audio-recorder.service';\r\nimport { SpeechToTextServiceFactory } from '../../services/speech-to-text.service';\r\nimport { LLMServiceFactory } from '../../services/llm.service';\r\nimport { SchemaConverter } from '../../utils/schema-converter';\r\n\r\n@Component({\r\n tag: 'voice-input-module',\r\n styleUrl: 'voice-input-module.css',\r\n shadow: true,\r\n})\r\nexport class VoiceFormRecorder {\r\n @Prop() formJson: string = '{}';\r\n @Prop() serviceConfig: string = '{}';\r\n @Prop() apiKey: string;\r\n @Prop() context: 'track'|'ng'|'ecoll-veto'|'ecoteka' = undefined;\r\n @Prop() classificationRootUrl: string = 'http://localhost';\r\n @Prop() language: 'fr'|'en' = 'en';\r\n @Prop() voiceOrOcr: 'voice'|'ocr'|'both' = undefined;\r\n \r\n @Prop() debug: boolean = false;\r\n @Prop() renderForm: boolean = false;\r\n @Prop() displayStatus: boolean = false;\r\n \r\n @Event() formFilled: EventEmitter<VoiceFormRecorderResult>;\r\n @Event() recordingStateChanged: EventEmitter<{ isRecording: boolean; state: string }>;\r\n\r\n @State() isRecording: boolean = false;\r\n @State() isProcessing: boolean = false;\r\n @State() statusMessage: string;\r\n @State() hasError: boolean = false;\r\n @State() transcription: string = '';\r\n @State() filledData: FilledFormData | null = null;\r\n @State() debugInfo: any = {};\r\n @State() isReadonlyMode: boolean = true; // Start in readonly preview mode\r\n\r\n private audioRecorder: AudioRecorderService;\r\n private speechToTextService: any;\r\n private llmService: any;\r\n private parsedSchema: any; // Support both simple FormSchema and complex nested schema\r\n private parsedConfig: ServiceProviderConfig;\r\n\r\n constructor() {\r\n this.audioRecorder = new AudioRecorderService();\r\n }\r\n\r\n componentWillLoad() {\r\n this.initializeServices();\r\n }\r\n\r\n @Watch('formJson')\r\n @Watch('serviceConfig')\r\n initializeServices() {\r\n try {\r\n if (!this.context){\r\n this.hasError = true;\r\n const errorMessage = (this.language == 'en' ? `Initialization error: context is '${this.context}'` : `Erreur d'initialisation: le contexte est '${this.context}'`);\r\n this.statusMessage = errorMessage;\r\n this.updateDebugInfo(errorMessage, { error: errorMessage });\r\n }\r\n else{\r\n // Parse form schema\r\n this.parsedSchema = JSON.parse(this.formJson || '{}');\r\n \r\n // Parse service configuration\r\n this.parsedConfig = JSON.parse(this.serviceConfig || '{}');\r\n \r\n // Add API key to config if provided via prop\r\n if (this.apiKey) {\r\n this.parsedConfig = {\r\n ...this.parsedConfig,\r\n speechToText: { ...this.parsedConfig.speechToText, apiKey: this.apiKey },\r\n llm: { ...this.parsedConfig.llm, apiKey: this.apiKey }\r\n };\r\n }\r\n \r\n // Initialize services\r\n this.speechToTextService = SpeechToTextServiceFactory.create(this.parsedConfig);\r\n this.llmService = LLMServiceFactory.create(this.parsedConfig);\r\n \r\n this.updateDebugInfo('Initialized', {\r\n schema: this.parsedSchema,\r\n config: this.parsedConfig\r\n });\r\n \r\n this.hasError = false;\r\n this.statusMessage = (this.language == 'en' ? 'Click to start recording' : 'Cliquer pour enregistrer');\r\n }\r\n } catch (error) {\r\n this.hasError = true;\r\n this.statusMessage = (this.language == 'en' ? `Initialization error: ${error.message}` : `Erreur d'initialisation: ${error.message}`);\r\n this.updateDebugInfo('Initialization Error', { error: error.message });\r\n }\r\n }\r\n\r\n private updateDebugInfo(action: string, data: any) {\r\n if (this.debug) {\r\n this.debugInfo = {\r\n ...this.debugInfo,\r\n [action]: {\r\n timestamp: new Date().toISOString(),\r\n data\r\n }\r\n };\r\n }\r\n }\r\n\r\n private async handleRecordClick() {\r\n if (this.isProcessing) return;\r\n\r\n if (this.isRecording) {\r\n await this.stopRecordingAndProcess();\r\n } else {\r\n await this.startRecording();\r\n }\r\n }\r\n\r\n private async startRecording() {\r\n try {\r\n this.hasError = false;\r\n this.statusMessage = (this.language == 'en' ? 'Starting recording...' : `Enregistrement ...`);\r\n this.updateDebugInfo('Start Recording Attempt', {});\r\n\r\n await this.audioRecorder.startRecording();\r\n \r\n this.isRecording = true;\r\n this.statusMessage = (this.language == 'en' ? 'Recording... Click to stop' : 'Enregistrement ... Cliquer pour stopper');\r\n this.updateDebugInfo('Recording Started', {});\r\n \r\n this.recordingStateChanged.emit({ \r\n isRecording: true, \r\n state: 'recording' \r\n });\r\n } catch (error) {\r\n this.hasError = true;\r\n this.statusMessage = (this.language == 'en' ? `Recording failed: ${error.message}` : `Echec de l'enregistrement : ${error.message}`);\r\n this.updateDebugInfo('Recording Error', { error: error.message });\r\n \r\n this.formFilled.emit({\r\n success: false,\r\n error: error.message\r\n });\r\n }\r\n }\r\n\r\n private async processJsonForm(jsonForm: string){\r\n // console.log(\"processJsonForm\", jsonForm);\r\n try {\r\n this.isProcessing = true;\r\n this.statusMessage = (this.language == 'en' ? 'Processing json...' : `Traitement du json ...`);\r\n\r\n // Fill form using LLM\r\n this.statusMessage = (this.language == 'en' ? 'Filling form fields...' : 'Remplissage du formulaire ...');\r\n const trimmedSchema = this.trimSchemaForAI(this.parsedSchema);\r\n const filledSchema = await this.llmService.fillFormFromJson(jsonForm, trimmedSchema);\r\n \r\n // Extract filled data\r\n this.filledData = this.extractFilledData(filledSchema);\r\n this.updateDebugInfo('Form Filled', { \r\n filledSchema, \r\n extractedData: this.filledData \r\n });\r\n \r\n this.parsedSchema = this.filledData;\r\n\r\n this.statusMessage = (this.language == 'en' ? 'Form completed!' : 'Formulaire remplis !');\r\n this.hasError = false;\r\n\r\n // Emit success event\r\n this.formFilled.emit({\r\n success: true,\r\n data: this.filledData,\r\n jsonForm: jsonForm\r\n });\r\n\r\n } catch (error) {\r\n this.hasError = true;\r\n this.statusMessage = (this.language == 'en' ? `Processing failed: ${error.message}` : `Erreur de traitement : ${error.message}`);\r\n \r\n this.formFilled.emit({\r\n success: false,\r\n error: error.message,\r\n jsonForm: jsonForm\r\n });\r\n } finally {\r\n this.isProcessing = false;\r\n }\r\n }\r\n\r\n private async stopRecordingAndProcess() {\r\n try {\r\n this.isRecording = false;\r\n this.isProcessing = true;\r\n this.statusMessage = (this.language == 'en' ? 'Processing audio...' : `Traitement de l'audio ...`);\r\n this.updateDebugInfo('Stop Recording', {});\r\n\r\n this.recordingStateChanged.emit({ \r\n isRecording: false, \r\n state: 'processing' \r\n });\r\n\r\n // Stop recording and get audio blob\r\n const audioBlob = await this.audioRecorder.stopRecording();\r\n this.updateDebugInfo('Audio Captured', { \r\n size: audioBlob.size, \r\n type: audioBlob.type \r\n });\r\n\r\n // Transcribe audio\r\n this.statusMessage = (this.language == 'en' ? 'Transcribing speech...' : 'Transcription du texte ...');\r\n const transcription = await this.speechToTextService.transcribe(audioBlob, this.language);\r\n this.transcription = transcription;\r\n this.updateDebugInfo('Transcription Complete', { transcription });\r\n\r\n if (!transcription.trim()) {\r\n throw new Error('No speech detected in the recording');\r\n }\r\n\r\n // Fill form using LLM\r\n this.statusMessage = (this.language == 'en' ? 'Filling form fields...' : 'Remplissage du formulaire ...');\r\n const trimmedSchema = this.trimSchemaForAI(this.parsedSchema);\r\n const filledSchema = await this.llmService.fillFormFromTranscription(transcription, trimmedSchema);\r\n \r\n // Extract filled data\r\n this.filledData = this.extractFilledData(filledSchema);\r\n this.updateDebugInfo('Form Filled', { \r\n filledSchema, \r\n extractedData: this.filledData \r\n });\r\n \r\n this.parsedSchema = this.filledData;\r\n\r\n this.statusMessage = (this.language == 'en' ? 'Form completed!' : 'Formulaire remplis !');\r\n this.hasError = false;\r\n\r\n // Emit success event\r\n this.formFilled.emit({\r\n success: true,\r\n data: this.filledData,\r\n transcription: transcription\r\n });\r\n\r\n } catch (error) {\r\n this.hasError = true;\r\n this.statusMessage = (this.language == 'en' ? `Processing failed: ${error.message}` : `Erreur de traitement : ${error.message}`);\r\n this.updateDebugInfo('Processing Error', { error: error.message });\r\n \r\n this.formFilled.emit({\r\n success: false,\r\n error: error.message,\r\n transcription: this.transcription\r\n });\r\n } finally {\r\n this.isProcessing = false;\r\n this.recordingStateChanged.emit({ \r\n isRecording: false, \r\n state: 'idle' \r\n });\r\n }\r\n }\r\n\r\n private extractFilledData(filledData: any): any {\r\n // console.log(\"extractFilledData\", filledData);\r\n const updatedSchema = JSON.parse(JSON.stringify(this.parsedSchema));\r\n switch(this.context){\r\n case \"ng\": \r\n if (filledData?.fields) {\r\n // Map AI response back to original schema structure\r\n filledData.fields.forEach((field: any) => {\r\n const originalField = updatedSchema.Children.find((child: any) => \r\n child.System_Name === field.name\r\n );\r\n \r\n if (originalField && field.value !== undefined && field.value !== null && field.value !== '') {\r\n if (!originalField.Settings) originalField.Settings = {};\r\n originalField.Settings.Default_Value = field.value;\r\n }\r\n });\r\n }\r\n \r\n break;\r\n case \"ecoll-veto\":\r\n // console.log(\"TODO extractFilledData\", filledData, updatedSchema);\r\n if (filledData?.fields) {\r\n // Map AI response back to original schema structure\r\n filledData.fields.forEach((field: any) => {\r\n let originalField = updatedSchema[0].items.find((child: any) => \r\n child.label === field.name\r\n );\r\n if (!originalField)\r\n originalField = updatedSchema[1].items.find((child: any) =>\r\n child.label === field.name\r\n );\r\n \r\n if (originalField && field.value !== undefined && field.value !== null && field.value !== '') {\r\n updatedSchema[2][originalField.name] = field.value;\r\n }\r\n });\r\n }\r\n break;\r\n case \"ecoteka\":\r\n // console.log(\"TODO extractFilledData\", filledData);\r\n break;\r\n case \"track\":\r\n default:\r\n const data: FilledFormData = {};\r\n \r\n Object.entries(filledData.fields).forEach(([fieldID, field]: [string, any]) => {\r\n if (field.default !== undefined && field.default !== null && field.default !== '') {\r\n updatedSchema[fieldID] = field.default;\r\n }\r\n if (field.value !== undefined && field.value !== null && field.value !== '') {\r\n for (const key in updatedSchema.fields) {\r\n const schemaField = updatedSchema.fields[key];\r\n if (schemaField.title === field.name) {\r\n schemaField.value = field.value || field.default;\r\n schemaField.default = field.value || field.default;\r\n break; // stop after finding the first match\r\n }\r\n }\r\n }\r\n });\r\n\r\n break;\r\n }\r\n \r\n // console.log(\"extractFilledData result\", updatedSchema);\r\n return updatedSchema;\r\n }\r\n\r\n private trimSchemaForAI(schema: any): any {\r\n // console.log(\"trimSchemaForAI\", schema);\r\n switch(this.context){\r\n case 'ng':\r\n const trimmed = { fields: [] };\r\n schema.Children.forEach((child: any) => {\r\n if (!child.System_Name || !child.Type) return;\r\n \r\n const fieldData: any = {\r\n name: child.System_Name,\r\n type: this.mapFieldType(child.Type)\r\n };\r\n \r\n // Add options for classification/select fields\r\n if (child.Type === 'InputClassification' && child.Children && child.Children.length > 0) {\r\n fieldData.options = child.Children.map((option: any) => \r\n option.System_Name || option.Label || option.toString()\r\n );\r\n }\r\n \r\n trimmed.fields.push(fieldData);\r\n });\r\n \r\n // console.log(\"Schema apres transformation, contexte NG:\", trimmed);\r\n return trimmed;\r\n case 'ecoll-veto':\r\n // console.log(\"TODO trimSchemaForAI\", schema)\r\n const mergedItemsSchema = (this.parsedSchema[0].items).concat(this.parsedSchema[1].items);\r\n\r\n if (mergedItemsSchema) {\r\n const trimmedSchema: FormSchema = {\r\n title: 'Form Name',\r\n description: 'Form Description',\r\n schema: {}\r\n };\r\n\r\n Object.entries(mergedItemsSchema).forEach(([key, field]: [string, any]) => {\r\n const fieldName = field.name;\r\n const fieldType = this.mapFieldType(field.type) as any;\r\n const fieldLabel = field.label || fieldName;\r\n const isRequired = field.required || false;\r\n const fieldValue = ''; //TODO\r\n \r\n trimmedSchema.schema[fieldName] = {\r\n type: fieldType,\r\n title: fieldLabel,\r\n options: field.options,\r\n readonly: field.readonly === true,\r\n default: '',\r\n };\r\n });\r\n\r\n // console.log(\"Schema apres transformation, contexte Track:\", trimmedSchema);\r\n return trimmedSchema;\r\n }\r\n case \"ecoteka\":\r\n // console.log(\"TODO trimSchemaForAI\", schema)\r\n break;\r\n case 'track':\r\n default:\r\n // Handle simple schema format (backward compatibility)\r\n if (schema?.schema ?? schema?.fields) {\r\n const trimmedSchema: FormSchema = {\r\n title: schema.title,\r\n description: schema.description,\r\n schema: {}\r\n };\r\n\r\n const finalSchema = schema?.schema ?? schema?.fields;\r\n\r\n Object.entries(finalSchema).forEach(([fieldName, field]: [string, any]) => {\r\n trimmedSchema.schema[fieldName] = {\r\n type: field.type,\r\n title: field.title,\r\n options: field.options,\r\n readonly: field.Enabled === false,\r\n default: field.DefaultValue,\r\n pattern: field.Mask,\r\n min: field.ValidationMin,\r\n max: field.ValidationMax\r\n };\r\n });\r\n\r\n \r\n // console.log(\"Schema apres transformation, contexte Track:\", trimmedSchema);\r\n return trimmedSchema;\r\n }\r\n break;\r\n }\r\n \r\n return schema;\r\n }\r\n\r\n private mapFieldType(type: string): string {\r\n const typeMapping = {\r\n 'InputAutocomplete': 'string',\r\n 'InputInteger': 'number',\r\n 'InputTextArea': 'string',\r\n 'InputDateTimePicker': 'datetime',\r\n 'InputDecimal': 'number',\r\n 'InputClassification': 'select',\r\n 'InputCheckbox': 'boolean',\r\n 'InputTextTranslation': 'string',\r\n 'thesaurus': 'string',\r\n 'position': 'string',\r\n 'text': 'string',\r\n 'textarea': 'string',\r\n 'number': 'number',\r\n 'date': 'date',\r\n 'datetime': 'datetime'\r\n };\r\n \r\n return typeMapping[type] || 'string';\r\n }\r\n\r\n // Utility methods exposed as public API\r\n @Method() public async convertXmlToJson(xmlForm: string): Promise<FormSchemaFieldsOnly> {\r\n return SchemaConverter.convertXmlToJson(xmlForm, this.classificationRootUrl, this.language);\r\n }\r\n\r\n @Method() public async convertJsonToXml(jsonForm: FormSchemaFieldsOnlyExtended): Promise<string> {\r\n return SchemaConverter.convertJsonToXml(jsonForm);\r\n }\r\n\r\n // Utility methods exposed as public API\r\n @Method() public async convertXmlToJsonLegacy(xmlForm: string): Promise<FormSchema> {\r\n return SchemaConverter.convertXmlToJsonLegacy(xmlForm);\r\n }\r\n\r\n @Method() public async convertJsonToXmlLegacy(jsonForm: FormSchema): Promise<string> {\r\n return SchemaConverter.convertJsonToXmlLegacy(jsonForm);\r\n }\r\n\r\n private renderUploadButton() {\r\n if (!['ocr', 'both'].includes(this.voiceOrOcr))\r\n return;\r\n return(\r\n <file-uploader\r\n batch={false}\r\n callback={(data) => {this.processJsonForm(data);}}\r\n >\r\n </file-uploader>\r\n );\r\n }\r\n\r\n private renderRecordButton() {\r\n if (!['voice', 'both', undefined].includes(this.voiceOrOcr))\r\n return;\r\n\r\n const buttonClass = [\r\n 'record-button',\r\n this.isRecording && 'recording',\r\n this.isProcessing && 'processing'\r\n ].filter(Boolean).join(' ');\r\n\r\n const isDisabled = this.isProcessing || this.hasError;\r\n\r\n return (\r\n <button \r\n class={buttonClass}\r\n onClick={() => this.handleRecordClick()}\r\n disabled={isDisabled}\r\n aria-label={this.isRecording ? 'Stop recording' : 'Start recording'}\r\n >\r\n {this.isProcessing ? (\r\n <svg class=\"record-icon\" viewBox=\"0 0 24 24\">\r\n <circle cx=\"12\" cy=\"12\" r=\"3\">\r\n <animate attributeName=\"r\" values=\"3;6;3\" dur=\"1s\" repeatCount=\"indefinite\" />\r\n <animate attributeName=\"opacity\" values=\"1;0.3;1\" dur=\"1s\" repeatCount=\"indefinite\" />\r\n </circle>\r\n </svg>\r\n ) : this.isRecording ? (\r\n <svg class=\"record-icon\" viewBox=\"0 0 24 24\">\r\n <rect x=\"6\" y=\"6\" width=\"12\" height=\"12\" rx=\"2\" />\r\n </svg>\r\n ) : (\r\n <svg class=\"record-icon\" viewBox=\"0 0 24 24\">\r\n <circle cx=\"12\" cy=\"12\" r=\"8\" />\r\n </svg>\r\n )}\r\n </button>\r\n );\r\n }\r\n\r\n private renderStatusMessage() {\r\n const statusClass = [\r\n 'status-text',\r\n this.hasError && 'error',\r\n this.filledData && !this.hasError && 'success'\r\n ].filter(Boolean).join(' ');\r\n\r\n return <div class={statusClass}>{this.statusMessage}</div>;\r\n }\r\n\r\n private renderFormPreview() {\r\n if (!this.parsedSchema) return null;\r\n\r\n const isPreview = this.isReadonlyMode && !this.filledData;\r\n const title = isPreview ? 'Form Preview (Voice input to fill)' : 'Voice-Filled Form:';\r\n \r\n return (\r\n <div class=\"form-preview\">\r\n <div class=\"form-preview-title\">{title}</div>\r\n <form class=\"voice-filled-form\">\r\n {this.renderFormFields()}\r\n </form>\r\n </div>\r\n );\r\n }\r\n\r\n private renderFormFields() {\r\n // console.log(\"renderFormFields\", this.parsedSchema);\r\n if (!this.parsedSchema) return null;\r\n \r\n switch(this.context){\r\n case \"ng\":\r\n return this.parsedSchema.Children.map((child: any) => {\r\n if (!child.System_Name || !child.Type) return null;\r\n \r\n const fieldName = child.System_Name;\r\n const fieldType = this.mapFieldType(child.Type);\r\n const fieldLabel = child.Settings?.Label || child.System_Name;\r\n const isRequired = child.Required || false;\r\n const fieldValue = child.Settings?.Default_Value;\r\n \r\n return (\r\n <div class=\"form-group\" key={fieldName}>\r\n <label htmlFor={fieldName} class=\"form-label\">\r\n {fieldLabel}\r\n {isRequired && <span class=\"required\">*</span>}\r\n </label>\r\n {this.renderFormField(fieldName, {\r\n type: fieldType,\r\n title: fieldLabel,\r\n required: isRequired,\r\n options: child.Children?.map((option: any) => \r\n option.System_Name || option.Label || option.toString()\r\n )\r\n }, fieldValue)}\r\n </div>\r\n );\r\n }).filter(Boolean);\r\n case \"ecoll-veto\":\r\n // NOTE STEP 2\r\n // console.log(\"TODO renderFormFields\", this.parsedSchema);\r\n const mergedItemsSchema = (this.parsedSchema[0].items).concat(this.parsedSchema[1].items);\r\n return Object.entries(mergedItemsSchema).map(([key, field]: [string, any]) => {\r\n \r\n const fieldName = field.name;\r\n field.type = this.mapFieldType(field.type);\r\n const fieldLabel = field.label || fieldName;\r\n const isRequired = field.required || false;\r\n const fieldValue = this.parsedSchema[2][fieldName];\r\n\r\n return (\r\n <div class=\"form-group\" key={fieldName}>\r\n <label htmlFor={fieldName} class=\"form-label\">\r\n { fieldLabel }\r\n {isRequired && <span class=\"required\">*</span>}\r\n </label>\r\n {this.renderFormField(fieldName, field, (this.filledData?.[fieldName] ?? fieldValue))}\r\n </div>\r\n );\r\n });\r\n case \"ecoteka\":\r\n // console.log(\"TODO renderFormFields\", this.parsedSchema);\r\n break;\r\n case \"track\":\r\n default:\r\n return Object.entries(this.parsedSchema.fields).map(([fieldName, field]: [string, any]) => (\r\n <div class=\"form-group\" key={fieldName}>\r\n <label htmlFor={fieldName} class=\"form-label\">\r\n {field.title || fieldName}\r\n {field.required && <span class=\"required\">*</span>}\r\n </label>\r\n {this.renderFormField(fieldName, field, (this.filledData?.[fieldName] ?? field.value))}\r\n </div>\r\n ));\r\n }\r\n \r\n return null;\r\n }\r\n\r\n private renderFormField(fieldName: string, field: any, value: any) {\r\n const isReadonly = this.isReadonlyMode && !this.filledData;\r\n const commonProps = {\r\n id: fieldName,\r\n name: fieldName,\r\n class: 'form-input',\r\n required: field.required,\r\n disabled: isReadonly\r\n };\r\n\r\n switch (field.type) {\r\n case 'select':\r\n if (isReadonly) {\r\n // In readonly mode, show all options as a list instead of dropdown\r\n return (\r\n <div class=\"readonly-select\">\r\n <div class=\"select-placeholder\">Available options:</div>\r\n <ul class=\"select-options-list\">\r\n {field.options?.map(option => (\r\n <li class=\"select-option\">{option}</li>\r\n ))}\r\n </ul>\r\n </div>\r\n );\r\n }\r\n return (\r\n <select id={fieldName} name={fieldName} class=\"form-input\" required={field.required}>\r\n <option value=\"\">-- Select --</option>\r\n {field.options?.map(option => (\r\n <option value={option} selected={value === option}>\r\n {option}\r\n </option>\r\n ))}\r\n </select>\r\n );\r\n \r\n case 'boolean':\r\n return (\r\n <input \r\n {...commonProps}\r\n type=\"checkbox\" \r\n class=\"form-checkbox\"\r\n checked={value === true || value === 'true'}\r\n />\r\n );\r\n \r\n case 'number':\r\n return (\r\n <input \r\n {...commonProps}\r\n type=\"number\" \r\n min={field.min}\r\n max={field.max}\r\n step=\"any\"\r\n value={value || ''}\r\n />\r\n );\r\n \r\n case 'date':\r\n return (\r\n <input \r\n {...commonProps}\r\n type='date'\r\n value={value ? (([d, m, y]) => `${y}-${m}-${d}`)(value.split(\"/\")) : ''}\r\n />\r\n );\r\n\r\n case 'datetime':\r\n return (\r\n <input \r\n {...commonProps}\r\n type='datetime-local'\r\n value={value || ''}\r\n />\r\n );\r\n \r\n default: // string\r\n return (\r\n <input \r\n {...commonProps}\r\n type=\"text\" \r\n pattern={field.pattern}\r\n placeholder={field.description}\r\n value={value || ''}\r\n />\r\n );\r\n }\r\n }\r\n\r\n private renderDebugPanel() {\r\n if (!this.debug) return null;\r\n\r\n return (\r\n <div class=\"debug-panel\">\r\n <div class=\"debug-title\">Debug Information:</div>\r\n <div class=\"debug-content\">\r\n {JSON.stringify(this.debugInfo, null, 2)}\r\n </div>\r\n </div>\r\n );\r\n }\r\n\r\n render() {\r\n return (\r\n <div>\r\n <div class={\"voice-recorder-container\" + (this.debug || this.renderForm ? \"-debug\" : \"\")}>\r\n {this.renderRecordButton()}\r\n {this.displayStatus ? this.renderStatusMessage() : \"\"}\r\n {this.renderUploadButton()}\r\n {this.renderForm ? this.renderFormPreview() : \"\"}\r\n {this.debug ? this.renderDebugPanel() : \"\"}\r\n </div>\r\n </div>\r\n );\r\n }\r\n}\r\n"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview entry point for your component library
|
|
3
|
+
*
|
|
4
|
+
* This is the entry point for your component library. Use this file to export utilities,
|
|
5
|
+
* constants or data structure that accompany your components.
|
|
6
|
+
*
|
|
7
|
+
* DO NOT use this file to export your components. Instead, use the recommended approaches
|
|
8
|
+
* to consume components of this package as outlined in the `README.md`.
|
|
9
|
+
*/
|
|
10
|
+
// export { format } from './utils/utils';
|
|
11
|
+
// export type * from './components.d.ts';
|
|
12
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,0CAA0C;AAC1C,0CAA0C","sourcesContent":["/**\r\n * @fileoverview entry point for your component library\r\n *\r\n * This is the entry point for your component library. Use this file to export utilities,\r\n * constants or data structure that accompany your components.\r\n *\r\n * DO NOT use this file to export your components. Instead, use the recommended approaches\r\n * to consume components of this package as outlined in the `README.md`.\r\n */\r\n\r\n// export { format } from './utils/utils';\r\n// export type * from './components.d.ts';\r\n"]}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
export class AudioRecorderService {
|
|
2
|
+
constructor() {
|
|
3
|
+
this.mediaRecorder = null;
|
|
4
|
+
this.audioChunks = [];
|
|
5
|
+
this.stream = null;
|
|
6
|
+
}
|
|
7
|
+
async startRecording() {
|
|
8
|
+
try {
|
|
9
|
+
// Check if the API exists before calling
|
|
10
|
+
if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) {
|
|
11
|
+
console.error('Failed to start recording:', 'Microphone access is not supported in this browser or the page is not served over HTTPS/localhost.');
|
|
12
|
+
return; // Exit gracefully instead of throwing
|
|
13
|
+
}
|
|
14
|
+
this.stream = await navigator.mediaDevices.getUserMedia({
|
|
15
|
+
audio: {
|
|
16
|
+
echoCancellation: true,
|
|
17
|
+
noiseSuppression: true,
|
|
18
|
+
autoGainControl: true
|
|
19
|
+
}
|
|
20
|
+
});
|
|
21
|
+
this.audioChunks = [];
|
|
22
|
+
this.mediaRecorder = new MediaRecorder(this.stream, {
|
|
23
|
+
mimeType: 'audio/webm;codecs=opus'
|
|
24
|
+
});
|
|
25
|
+
this.mediaRecorder.ondataavailable = (event) => {
|
|
26
|
+
if (event.data && event.data.size > 0) {
|
|
27
|
+
this.audioChunks.push(event.data);
|
|
28
|
+
}
|
|
29
|
+
};
|
|
30
|
+
this.mediaRecorder.start(100); // Collect data every 100ms
|
|
31
|
+
}
|
|
32
|
+
catch (error) {
|
|
33
|
+
console.error('Failed to start recording:', error);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
async stopRecording() {
|
|
37
|
+
return new Promise((resolve, reject) => {
|
|
38
|
+
if (!this.mediaRecorder) {
|
|
39
|
+
reject(new Error('No active recording found'));
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
this.mediaRecorder.onstop = () => {
|
|
43
|
+
const audioBlob = new Blob(this.audioChunks, { type: 'audio/webm' });
|
|
44
|
+
this.cleanup();
|
|
45
|
+
resolve(audioBlob);
|
|
46
|
+
};
|
|
47
|
+
this.mediaRecorder.onerror = (event) => {
|
|
48
|
+
reject(new Error(`Recording error: ${event}`));
|
|
49
|
+
};
|
|
50
|
+
this.mediaRecorder.stop();
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
isRecording() {
|
|
54
|
+
var _a;
|
|
55
|
+
return ((_a = this.mediaRecorder) === null || _a === void 0 ? void 0 : _a.state) === 'recording';
|
|
56
|
+
}
|
|
57
|
+
cleanup() {
|
|
58
|
+
if (this.stream) {
|
|
59
|
+
this.stream.getTracks().forEach(track => track.stop());
|
|
60
|
+
this.stream = null;
|
|
61
|
+
}
|
|
62
|
+
this.mediaRecorder = null;
|
|
63
|
+
this.audioChunks = [];
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
//# sourceMappingURL=audio-recorder.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audio-recorder.service.js","sourceRoot":"","sources":["../../src/services/audio-recorder.service.ts"],"names":[],"mappings":"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","sourcesContent":["export class AudioRecorderService {\r\n private mediaRecorder: MediaRecorder | null = null;\r\n private audioChunks: Blob[] = [];\r\n private stream: MediaStream | null = null;\r\n\r\n async startRecording(): Promise<void> {\r\n try {\r\n // Check if the API exists before calling\r\n if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) {\r\n console.error(\r\n 'Failed to start recording:', 'Microphone access is not supported in this browser or the page is not served over HTTPS/localhost.'\r\n );\r\n return; // Exit gracefully instead of throwing\r\n }\r\n \r\n this.stream = await navigator.mediaDevices.getUserMedia({\r\n audio: {\r\n echoCancellation: true,\r\n noiseSuppression: true,\r\n autoGainControl: true\r\n }\r\n });\r\n\r\n this.audioChunks = [];\r\n this.mediaRecorder = new MediaRecorder(this.stream, {\r\n mimeType: 'audio/webm;codecs=opus'\r\n });\r\n\r\n this.mediaRecorder.ondataavailable = (event) => {\r\n if (event.data && event.data.size > 0) {\r\n this.audioChunks.push(event.data);\r\n }\r\n };\r\n\r\n this.mediaRecorder.start(100); // Collect data every 100ms\r\n } catch (error: any) {\r\n console.error('Failed to start recording:', error);\r\n }\r\n }\r\n\r\n async stopRecording(): Promise<Blob> {\r\n return new Promise((resolve, reject) => {\r\n if (!this.mediaRecorder) {\r\n reject(new Error('No active recording found'));\r\n return;\r\n }\r\n\r\n this.mediaRecorder.onstop = () => {\r\n const audioBlob = new Blob(this.audioChunks, { type: 'audio/webm' });\r\n this.cleanup();\r\n resolve(audioBlob);\r\n };\r\n\r\n this.mediaRecorder.onerror = (event) => {\r\n reject(new Error(`Recording error: ${event}`));\r\n };\r\n\r\n this.mediaRecorder.stop();\r\n });\r\n }\r\n\r\n isRecording(): boolean {\r\n return this.mediaRecorder?.state === 'recording';\r\n }\r\n\r\n private cleanup(): void {\r\n if (this.stream) {\r\n this.stream.getTracks().forEach(track => track.stop());\r\n this.stream = null;\r\n }\r\n this.mediaRecorder = null;\r\n this.audioChunks = [];\r\n }\r\n}\r\n"]}
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
export class OpenAILLMService {
|
|
2
|
+
constructor(config) {
|
|
3
|
+
// Get API key from config or environment
|
|
4
|
+
this.apiKey = (config === null || config === void 0 ? void 0 : config.apiKey) || this.getEnvironmentVariable('OPENAI_API_KEY') || '';
|
|
5
|
+
// the newest OpenAI model is "gpt-4.1-mini". do not change this unless explicitly requested by the user
|
|
6
|
+
// this.model = config?.model || 'gpt-4.1-mini';
|
|
7
|
+
this.model = (config === null || config === void 0 ? void 0 : config.model) || 'gpt-4.1';
|
|
8
|
+
this.baseUrl = (config === null || config === void 0 ? void 0 : config.baseUrl) || 'https://api.openai.com/v1';
|
|
9
|
+
if (!this.apiKey) {
|
|
10
|
+
throw new Error('OpenAI API key is required for LLM service');
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
getEnvironmentVariable(name) {
|
|
14
|
+
// In browser environment, we might get env vars through other means
|
|
15
|
+
if (typeof process !== 'undefined' && process.env) {
|
|
16
|
+
return process.env[name];
|
|
17
|
+
}
|
|
18
|
+
// Check if it's available as a global variable or through other means
|
|
19
|
+
return window[name] || undefined;
|
|
20
|
+
}
|
|
21
|
+
getOptimizeFieldsDescription(schema) {
|
|
22
|
+
return Object.values(schema).map((field) => {
|
|
23
|
+
var _a;
|
|
24
|
+
return `- ${(_a = field.name) !== null && _a !== void 0 ? _a : field.title} ` +
|
|
25
|
+
`(${field.type}` +
|
|
26
|
+
`${field.required ? ', required' : ''}` +
|
|
27
|
+
`${field.readonly ? ', readonly' : ''}` +
|
|
28
|
+
`${field.min && field.min !== "" ? ', min=' + field.min : ''}` +
|
|
29
|
+
`${field.max && field.max !== "" ? ', max=' + field.max : ''}` +
|
|
30
|
+
`)` +
|
|
31
|
+
`${field.options ? ` - options: ${field.options.join(', ')}` : ''}`;
|
|
32
|
+
}).join('\n');
|
|
33
|
+
}
|
|
34
|
+
async fillFormFromTranscription(transcription, schema) {
|
|
35
|
+
return this.fillForm(transcription, schema, true);
|
|
36
|
+
}
|
|
37
|
+
async fillFormFromJson(json, schema) {
|
|
38
|
+
return this.fillForm(json, schema, false);
|
|
39
|
+
}
|
|
40
|
+
async fillForm(data, schema, dataIsTranscription = true) {
|
|
41
|
+
var _a, _b;
|
|
42
|
+
try {
|
|
43
|
+
// Handle complex schema format with fields array
|
|
44
|
+
if ((schema === null || schema === void 0 ? void 0 : schema.fields) || (schema === null || schema === void 0 ? void 0 : schema.schema)) {
|
|
45
|
+
const finalSchema = (schema === null || schema === void 0 ? void 0 : schema.fields) || (schema === null || schema === void 0 ? void 0 : schema.schema);
|
|
46
|
+
const systemPrompt = `You are an expert form-filling assistant. You will receive a voice transcription and form field definitions. Your task is to extract values from the spoken content.
|
|
47
|
+
Rules:
|
|
48
|
+
1. Only extract values that can be confidently determined from the transcription
|
|
49
|
+
2. Respect field types (string, number, datetime, boolean, select)
|
|
50
|
+
3. For datetime fields, use ISO format (YYYY-MM-DDTHH:MM)
|
|
51
|
+
4. For date fields, use the following format: DD/MM/YYY
|
|
52
|
+
5. For boolean fields, interpret yes/no, true/false, positive/negative responses
|
|
53
|
+
6. For select fields, use exact option values from the provided choices
|
|
54
|
+
7. Only include fields where relevant information is found
|
|
55
|
+
8. The current GMT datetime is ${new Date().toGMTString()}
|
|
56
|
+
9. Respect the constraints written between parenthesis for readonly status (readonly fields must not be filled with values), min and max values, whatever the transcription says
|
|
57
|
+
|
|
58
|
+
Respond with JSON in this exact format: {"fields": [{"name": "field_name", "value": "extracted_value"}]}`;
|
|
59
|
+
let userPrompt = `
|
|
60
|
+
Voice transcription: "${data}"
|
|
61
|
+
|
|
62
|
+
Form fields:
|
|
63
|
+
${this.getOptimizeFieldsDescription(finalSchema)}
|
|
64
|
+
|
|
65
|
+
Please extract values from the Transcription for these fields.
|
|
66
|
+
`;
|
|
67
|
+
//TODO
|
|
68
|
+
// const userPrompt = (
|
|
69
|
+
// dataIsTranscription
|
|
70
|
+
// ?
|
|
71
|
+
// `Voice transcription: "${data}"`
|
|
72
|
+
// :
|
|
73
|
+
// `Json datas: "${JSON.stringify(data)}"`
|
|
74
|
+
// )+`
|
|
75
|
+
// Form fields:
|
|
76
|
+
// ${this.getOptimizeFieldsDescription(finalSchema)}
|
|
77
|
+
// Please extract values from the ` +
|
|
78
|
+
// (
|
|
79
|
+
// dataIsTranscription
|
|
80
|
+
// ?
|
|
81
|
+
// `Transcription`
|
|
82
|
+
// :
|
|
83
|
+
// `Json generated file`
|
|
84
|
+
// )+` for these fields.`;
|
|
85
|
+
const response = await fetch(`${this.baseUrl}/chat/completions`, {
|
|
86
|
+
method: 'POST',
|
|
87
|
+
headers: {
|
|
88
|
+
'Content-Type': 'application/json',
|
|
89
|
+
'Authorization': `Bearer ${this.apiKey}`,
|
|
90
|
+
},
|
|
91
|
+
body: JSON.stringify({
|
|
92
|
+
model: this.model,
|
|
93
|
+
messages: [
|
|
94
|
+
{ role: 'system', content: systemPrompt },
|
|
95
|
+
{ role: 'user', content: userPrompt }
|
|
96
|
+
],
|
|
97
|
+
response_format: { type: 'json_object' },
|
|
98
|
+
temperature: 0.1,
|
|
99
|
+
}),
|
|
100
|
+
});
|
|
101
|
+
if (!response.ok) {
|
|
102
|
+
const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));
|
|
103
|
+
throw new Error(`LLM API failed: ${((_a = errorData.error) === null || _a === void 0 ? void 0 : _a.message) || response.statusText}`);
|
|
104
|
+
}
|
|
105
|
+
const result = await response.json();
|
|
106
|
+
return JSON.parse(result.choices[0].message.content);
|
|
107
|
+
}
|
|
108
|
+
// Handle simple schema format (backward compatibility)
|
|
109
|
+
const systemPrompt = `You are an expert form-filling assistant. You will receive a voice transcription and a JSON form schema. Your task is to intelligently fill the form fields based on the spoken content.
|
|
110
|
+
Rules:
|
|
111
|
+
1. Only fill fields that can be confidently determined from the transcription
|
|
112
|
+
2. Respect field types (string, number, date, boolean, select)
|
|
113
|
+
3. For datetime fields, use ISO format (YYYY-MM-DDTHH:MM)
|
|
114
|
+
4. For date fields, use the following format: DD/MM/YYY
|
|
115
|
+
5. For boolean fields, interpret yes/no, true/false, positive/negative responses
|
|
116
|
+
6. For select fields, match the closest option from the provided choices
|
|
117
|
+
7. Leave fields empty if no relevant information is found
|
|
118
|
+
8. Return the same schema structure with 'default' values filled
|
|
119
|
+
9. The current GMT datetime is ${new Date().toGMTString()}
|
|
120
|
+
10. Respect the constraints written between parenthesis for readonly status (readonly fields must not be filled with values), min and max values, whatever the transcription says
|
|
121
|
+
|
|
122
|
+
Respond with JSON in this exact format: {"schema": {...}}`;
|
|
123
|
+
let userPrompt = `
|
|
124
|
+
Voice transcription: "${data}"
|
|
125
|
+
|
|
126
|
+
Form schema to fill:
|
|
127
|
+
${JSON.stringify(schema, null, 2)}
|
|
128
|
+
|
|
129
|
+
Please fill the form fields based on the Transcription and return the schema with default values populated.
|
|
130
|
+
`;
|
|
131
|
+
// const userPrompt = (
|
|
132
|
+
// dataIsTranscription
|
|
133
|
+
// ?
|
|
134
|
+
// `Voice transcription: "${data}"`
|
|
135
|
+
// :
|
|
136
|
+
// `Json datas: "${JSON.stringify(data)}"`
|
|
137
|
+
// )+`
|
|
138
|
+
// Form schema to fill:
|
|
139
|
+
// ${JSON.stringify(schema, null, 2)}
|
|
140
|
+
// Please fill the form fields based on the ` +
|
|
141
|
+
// (
|
|
142
|
+
// dataIsTranscription
|
|
143
|
+
// ?
|
|
144
|
+
// `Transcription`
|
|
145
|
+
// :
|
|
146
|
+
// `Json generated file`
|
|
147
|
+
// )+` nd return the schema with default values populated.`;
|
|
148
|
+
const response = await fetch(`${this.baseUrl}/chat/completions`, {
|
|
149
|
+
method: 'POST',
|
|
150
|
+
headers: {
|
|
151
|
+
'Content-Type': 'application/json',
|
|
152
|
+
'Authorization': `Bearer ${this.apiKey}`,
|
|
153
|
+
},
|
|
154
|
+
body: JSON.stringify({
|
|
155
|
+
model: this.model,
|
|
156
|
+
messages: [
|
|
157
|
+
{ role: 'system', content: systemPrompt },
|
|
158
|
+
{ role: 'user', content: userPrompt }
|
|
159
|
+
],
|
|
160
|
+
response_format: { type: 'json_object' },
|
|
161
|
+
temperature: 0.1, // Low temperature for consistency
|
|
162
|
+
}),
|
|
163
|
+
});
|
|
164
|
+
if (!response.ok) {
|
|
165
|
+
const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));
|
|
166
|
+
throw new Error(`LLM API failed: ${((_b = errorData.error) === null || _b === void 0 ? void 0 : _b.message) || response.statusText}`);
|
|
167
|
+
}
|
|
168
|
+
const result = await response.json();
|
|
169
|
+
const filledSchema = JSON.parse(result.choices[0].message.content);
|
|
170
|
+
// Validate that the response has the correct structure
|
|
171
|
+
if (!filledSchema.schema) {
|
|
172
|
+
throw new Error('Invalid response format from LLM service');
|
|
173
|
+
}
|
|
174
|
+
return filledSchema;
|
|
175
|
+
}
|
|
176
|
+
catch (error) {
|
|
177
|
+
throw new Error(`Form filling failed: ${error.message}`);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
export class LLMServiceFactory {
|
|
182
|
+
static create(config) {
|
|
183
|
+
var _a;
|
|
184
|
+
const provider = ((_a = config.llm) === null || _a === void 0 ? void 0 : _a.provider) || 'openai';
|
|
185
|
+
switch (provider) {
|
|
186
|
+
case 'openai':
|
|
187
|
+
return new OpenAILLMService(config.llm);
|
|
188
|
+
default:
|
|
189
|
+
throw new Error(`Unsupported LLM provider: ${provider}`);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
//# sourceMappingURL=llm.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"llm.service.js","sourceRoot":"","sources":["../../src/services/llm.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { LLMProvider, ServiceProviderConfig } from '../types/service-providers.types';\r\nimport { FormSchema } from '../types/form-schema.types';\r\n\r\nexport class OpenAILLMService implements LLMProvider {\r\n private apiKey: string;\r\n private model: string;\r\n private baseUrl: string;\r\n\r\n constructor(config: ServiceProviderConfig['llm']) {\r\n // Get API key from config or environment\r\n this.apiKey = config?.apiKey || this.getEnvironmentVariable('OPENAI_API_KEY') || '';\r\n // the newest OpenAI model is \"gpt-4.1-mini\". do not change this unless explicitly requested by the user\r\n // this.model = config?.model || 'gpt-4.1-mini';\r\n this.model = config?.model || 'gpt-4.1';\r\n this.baseUrl = config?.baseUrl || 'https://api.openai.com/v1';\r\n \r\n if (!this.apiKey) {\r\n throw new Error('OpenAI API key is required for LLM service');\r\n }\r\n }\r\n\r\n private getEnvironmentVariable(name: string): string | undefined {\r\n // In browser environment, we might get env vars through other means\r\n if (typeof process !== 'undefined' && process.env) {\r\n return process.env[name];\r\n }\r\n // Check if it's available as a global variable or through other means\r\n return (window as any)[name] || undefined;\r\n }\r\n\r\n private getOptimizeFieldsDescription(schema: any){\r\n return Object.values(schema).map((field: any) => \r\n `- ${field.name ?? field.title} ` +\r\n `(${field.type}` +\r\n `${field.required ? ', required' : ''}` +\r\n `${field.readonly ? ', readonly' : ''}` +\r\n `${field.min && field.min !== \"\" ? ', min='+field.min : ''}` +\r\n `${field.max && field.max !== \"\" ? ', max='+field.max : ''}` +\r\n `)` +\r\n `${field.options ? ` - options: ${field.options.join(', ')}` : ''}`\r\n ).join('\\n')\r\n }\r\n\r\n async fillFormFromTranscription(transcription: string, schema: any): Promise<any> {\r\n return this.fillForm(transcription, schema, true);\r\n }\r\n\r\n async fillFormFromJson(json: string, schema: any): Promise<any> {\r\n return this.fillForm(json, schema, false);\r\n }\r\n\r\n private async fillForm(data: string, schema: any, dataIsTranscription: boolean = true): Promise<any> {\r\n try {\r\n // Handle complex schema format with fields array\r\n if (schema?.fields || schema?.schema) {\r\n const finalSchema = schema?.fields || schema?.schema;\r\n const systemPrompt = `You are an expert form-filling assistant. You will receive a voice transcription and form field definitions. Your task is to extract values from the spoken content.\r\n Rules:\r\n 1. Only extract values that can be confidently determined from the transcription\r\n 2. Respect field types (string, number, datetime, boolean, select)\r\n 3. For datetime fields, use ISO format (YYYY-MM-DDTHH:MM)\r\n 4. For date fields, use the following format: DD/MM/YYY\r\n 5. For boolean fields, interpret yes/no, true/false, positive/negative responses\r\n 6. For select fields, use exact option values from the provided choices\r\n 7. Only include fields where relevant information is found\r\n 8. The current GMT datetime is ${(new Date() as any).toGMTString()}\r\n 9. Respect the constraints written between parenthesis for readonly status (readonly fields must not be filled with values), min and max values, whatever the transcription says\r\n\r\n Respond with JSON in this exact format: {\"fields\": [{\"name\": \"field_name\", \"value\": \"extracted_value\"}]}`;\r\n\r\n let userPrompt = `\r\n Voice transcription: \"${data}\"\r\n\r\n Form fields:\r\n ${this.getOptimizeFieldsDescription(finalSchema)}\r\n\r\n Please extract values from the Transcription for these fields.\r\n `;\r\n\r\n //TODO\r\n\r\n // const userPrompt = (\r\n // dataIsTranscription \r\n // ?\r\n // `Voice transcription: \"${data}\"`\r\n // : \r\n // `Json datas: \"${JSON.stringify(data)}\"`\r\n // )+`\r\n\r\n // Form fields:\r\n // ${this.getOptimizeFieldsDescription(finalSchema)}\r\n\r\n // Please extract values from the ` + \r\n // (\r\n // dataIsTranscription \r\n // ?\r\n // `Transcription`\r\n // : \r\n // `Json generated file`\r\n // )+` for these fields.`;\r\n\r\n const response = await fetch(`${this.baseUrl}/chat/completions`, {\r\n method: 'POST',\r\n headers: {\r\n 'Content-Type': 'application/json',\r\n 'Authorization': `Bearer ${this.apiKey}`,\r\n },\r\n body: JSON.stringify({\r\n model: this.model,\r\n messages: [\r\n { role: 'system', content: systemPrompt },\r\n { role: 'user', content: userPrompt }\r\n ],\r\n response_format: { type: 'json_object' },\r\n temperature: 0.1,\r\n }),\r\n });\r\n\r\n if (!response.ok) {\r\n const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));\r\n throw new Error(`LLM API failed: ${errorData.error?.message || response.statusText}`);\r\n }\r\n\r\n const result = await response.json();\r\n return JSON.parse(result.choices[0].message.content);\r\n }\r\n\r\n // Handle simple schema format (backward compatibility)\r\n const systemPrompt = `You are an expert form-filling assistant. You will receive a voice transcription and a JSON form schema. Your task is to intelligently fill the form fields based on the spoken content.\r\n Rules:\r\n 1. Only fill fields that can be confidently determined from the transcription\r\n 2. Respect field types (string, number, date, boolean, select)\r\n 3. For datetime fields, use ISO format (YYYY-MM-DDTHH:MM)\r\n 4. For date fields, use the following format: DD/MM/YYY\r\n 5. For boolean fields, interpret yes/no, true/false, positive/negative responses\r\n 6. For select fields, match the closest option from the provided choices\r\n 7. Leave fields empty if no relevant information is found\r\n 8. Return the same schema structure with 'default' values filled\r\n 9. The current GMT datetime is ${(new Date() as any).toGMTString()}\r\n 10. Respect the constraints written between parenthesis for readonly status (readonly fields must not be filled with values), min and max values, whatever the transcription says\r\n\r\n Respond with JSON in this exact format: {\"schema\": {...}}`;\r\n\r\n let userPrompt = `\r\n Voice transcription: \"${data}\"\r\n\r\n Form schema to fill:\r\n ${JSON.stringify(schema, null, 2)}\r\n\r\n Please fill the form fields based on the Transcription and return the schema with default values populated.\r\n `;\r\n\r\n // const userPrompt = (\r\n // dataIsTranscription \r\n // ?\r\n // `Voice transcription: \"${data}\"`\r\n // : \r\n // `Json datas: \"${JSON.stringify(data)}\"`\r\n // )+`\r\n\r\n // Form schema to fill:\r\n // ${JSON.stringify(schema, null, 2)}\r\n\r\n // Please fill the form fields based on the ` + \r\n // (\r\n // dataIsTranscription \r\n // ?\r\n // `Transcription`\r\n // : \r\n // `Json generated file`\r\n // )+` nd return the schema with default values populated.`;\r\n\r\n const response = await fetch(`${this.baseUrl}/chat/completions`, {\r\n method: 'POST',\r\n headers: {\r\n 'Content-Type': 'application/json',\r\n 'Authorization': `Bearer ${this.apiKey}`,\r\n },\r\n body: JSON.stringify({\r\n model: this.model,\r\n messages: [\r\n { role: 'system', content: systemPrompt },\r\n { role: 'user', content: userPrompt }\r\n ],\r\n response_format: { type: 'json_object' },\r\n temperature: 0.1, // Low temperature for consistency\r\n }),\r\n });\r\n\r\n if (!response.ok) {\r\n const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));\r\n throw new Error(`LLM API failed: ${errorData.error?.message || response.statusText}`);\r\n }\r\n\r\n const result = await response.json();\r\n const filledSchema = JSON.parse(result.choices[0].message.content);\r\n \r\n // Validate that the response has the correct structure\r\n if (!filledSchema.schema) {\r\n throw new Error('Invalid response format from LLM service');\r\n }\r\n\r\n return filledSchema;\r\n } catch (error) {\r\n throw new Error(`Form filling failed: ${error.message}`);\r\n }\r\n }\r\n}\r\n\r\nexport class LLMServiceFactory {\r\n static create(config: ServiceProviderConfig): LLMProvider {\r\n const provider = config.llm?.provider || 'openai';\r\n \r\n switch (provider) {\r\n case 'openai':\r\n return new OpenAILLMService(config.llm);\r\n default:\r\n throw new Error(`Unsupported LLM provider: ${provider}`);\r\n }\r\n }\r\n}\r\n"]}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
export class WhisperSpeechToTextService {
|
|
2
|
+
constructor(config) {
|
|
3
|
+
// Get API key from config or environment
|
|
4
|
+
this.apiKey = (config === null || config === void 0 ? void 0 : config.apiKey) || this.getEnvironmentVariable('OPENAI_API_KEY') || '';
|
|
5
|
+
this.baseUrl = (config === null || config === void 0 ? void 0 : config.baseUrl) || 'https://api.openai.com/v1';
|
|
6
|
+
if (!this.apiKey) {
|
|
7
|
+
throw new Error('OpenAI API key is required for Whisper service');
|
|
8
|
+
}
|
|
9
|
+
}
|
|
10
|
+
getEnvironmentVariable(name) {
|
|
11
|
+
// In browser environment, we might get env vars through other means
|
|
12
|
+
if (typeof process !== 'undefined' && process.env) {
|
|
13
|
+
return process.env[name];
|
|
14
|
+
}
|
|
15
|
+
// Check if it's available as a global variable or through other means
|
|
16
|
+
return window[name] || undefined;
|
|
17
|
+
}
|
|
18
|
+
async transcribe(audioBlob, lang = 'en') {
|
|
19
|
+
var _a;
|
|
20
|
+
try {
|
|
21
|
+
const formData = new FormData();
|
|
22
|
+
// Convert webm to a format Whisper can handle
|
|
23
|
+
const audioFile = new File([audioBlob], 'audio.webm', { type: 'audio/webm' });
|
|
24
|
+
formData.append('file', audioFile);
|
|
25
|
+
formData.append('model', 'gpt-4o-transcribe'); // >>> tronque le texte ?
|
|
26
|
+
// formData.append('model', 'gpt-4o-mini-transcribe');// >>> tronque le texte ?
|
|
27
|
+
// formData.append('model', 'whisper-1');
|
|
28
|
+
formData.append('language', lang);
|
|
29
|
+
formData.append('response_format', 'json');
|
|
30
|
+
formData.append('max_output_tokens', '2000');
|
|
31
|
+
const response = await fetch(`${this.baseUrl}/audio/transcriptions`, {
|
|
32
|
+
method: 'POST',
|
|
33
|
+
headers: {
|
|
34
|
+
'Authorization': `Bearer ${this.apiKey}`,
|
|
35
|
+
},
|
|
36
|
+
body: formData,
|
|
37
|
+
});
|
|
38
|
+
if (!response.ok) {
|
|
39
|
+
const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));
|
|
40
|
+
throw new Error(`Transcription failed: ${((_a = errorData.error) === null || _a === void 0 ? void 0 : _a.message) || response.statusText}`);
|
|
41
|
+
}
|
|
42
|
+
const result = await response.json();
|
|
43
|
+
return result.text || '';
|
|
44
|
+
}
|
|
45
|
+
catch (error) {
|
|
46
|
+
throw new Error(`Speech-to-text transcription failed: ${error.message}`);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
export class SpeechToTextServiceFactory {
|
|
51
|
+
static create(config) {
|
|
52
|
+
var _a;
|
|
53
|
+
const provider = ((_a = config.speechToText) === null || _a === void 0 ? void 0 : _a.provider) || 'whisper';
|
|
54
|
+
switch (provider) {
|
|
55
|
+
case 'whisper':
|
|
56
|
+
return new WhisperSpeechToTextService(config.speechToText);
|
|
57
|
+
default:
|
|
58
|
+
throw new Error(`Unsupported speech-to-text provider: ${provider}`);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
//# sourceMappingURL=speech-to-text.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"speech-to-text.service.js","sourceRoot":"","sources":["../../src/services/speech-to-text.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { SpeechToTextProvider, ServiceProviderConfig } from '../types/service-providers.types';\r\n\r\nexport class WhisperSpeechToTextService implements SpeechToTextProvider {\r\n private apiKey: string;\r\n private baseUrl: string;\r\n\r\n constructor(config: ServiceProviderConfig['speechToText']) {\r\n // Get API key from config or environment\r\n this.apiKey = config?.apiKey || this.getEnvironmentVariable('OPENAI_API_KEY') || '';\r\n this.baseUrl = config?.baseUrl || 'https://api.openai.com/v1';\r\n \r\n if (!this.apiKey) {\r\n throw new Error('OpenAI API key is required for Whisper service');\r\n }\r\n }\r\n\r\n private getEnvironmentVariable(name: string): string | undefined {\r\n // In browser environment, we might get env vars through other means\r\n if (typeof process !== 'undefined' && process.env) {\r\n return process.env[name];\r\n }\r\n // Check if it's available as a global variable or through other means\r\n return (window as any)[name] || undefined;\r\n }\r\n\r\n async transcribe(audioBlob: Blob, lang: string = 'en'): Promise<string> {\r\n try {\r\n const formData = new FormData();\r\n \r\n // Convert webm to a format Whisper can handle\r\n const audioFile = new File([audioBlob], 'audio.webm', { type: 'audio/webm' });\r\n formData.append('file', audioFile);\r\n formData.append('model', 'gpt-4o-transcribe');// >>> tronque le texte ?\r\n // formData.append('model', 'gpt-4o-mini-transcribe');// >>> tronque le texte ?\r\n // formData.append('model', 'whisper-1');\r\n formData.append('language', lang);\r\n formData.append('response_format', 'json');\r\n formData.append('max_output_tokens', '2000');\r\n\r\n const response = await fetch(`${this.baseUrl}/audio/transcriptions`, {\r\n method: 'POST',\r\n headers: {\r\n 'Authorization': `Bearer ${this.apiKey}`,\r\n },\r\n body: formData,\r\n });\r\n\r\n if (!response.ok) {\r\n const errorData = await response.json().catch(() => ({ error: 'Unknown error' }));\r\n throw new Error(`Transcription failed: ${errorData.error?.message || response.statusText}`);\r\n }\r\n\r\n const result = await response.json();\r\n return result.text || '';\r\n } catch (error) {\r\n throw new Error(`Speech-to-text transcription failed: ${error.message}`);\r\n }\r\n }\r\n}\r\n\r\nexport class SpeechToTextServiceFactory {\r\n static create(config: ServiceProviderConfig): SpeechToTextProvider {\r\n const provider = config.speechToText?.provider || 'whisper';\r\n \r\n switch (provider) {\r\n case 'whisper':\r\n return new WhisperSpeechToTextService(config.speechToText);\r\n default:\r\n throw new Error(`Unsupported speech-to-text provider: ${provider}`);\r\n }\r\n }\r\n}\r\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"form-schema.types.js","sourceRoot":"","sources":["../../src/types/form-schema.types.ts"],"names":[],"mappings":"","sourcesContent":["export interface FormField {\r\n type: 'string' | 'number' | 'date' | 'boolean' | 'select' | 'header' | 'dbpicker';\r\n title: string;\r\n description?: string;\r\n required?: boolean;\r\n default?: any;\r\n options?: string[]; // For select fields\r\n pattern?: string; // For validation\r\n readonly?: boolean;\r\n min?: number;\r\n max?: number;\r\n}\r\nexport interface FormFieldExtended extends FormField {\r\n realType?: string;\r\n pickerOptions?: any[];\r\n\r\n ID?: string;\r\n TFie_PK_ID?: string;\r\n TVal_PK_ID?: string;\r\n TFIn_PK_ID?: string;\r\n Enabled?: boolean;\r\n isForm?: string;\r\n LabelText?: string;\r\n SystemName?: string;\r\n HelpText?: string;\r\n CssStyle?: string;\r\n ControlType?: string;\r\n ValidationRequired?: string;\r\n ValidationMin?: string;\r\n ValidationMax?: string;\r\n Mask?: string;\r\n TypeId?: string;\r\n Unit?: string;\r\n TFie_Fullpath?: string;\r\n TVal_FK_Parent_ID?: string;\r\n value?: any;\r\n}\r\n\r\nexport interface FormSchema {\r\n schema: {\r\n [fieldName: string]: FormField;\r\n };\r\n title?: string;\r\n description?: string;\r\n}\r\n\r\nexport interface FormSchemaFieldsOnly {\r\n fields: {\r\n [fieldName: string]: FormField;\r\n };\r\n}\r\n\r\nexport interface FormSchemaExtended {\r\n schema: {\r\n [fieldName: string]: FormFieldExtended;\r\n };\r\n title?: string;\r\n description?: string;\r\n}\r\n\r\n\r\nexport interface FormSchemaFieldsOnlyExtended {\r\n fields: {\r\n [fieldName: string]: FormFieldExtended;\r\n };\r\n}\r\n\r\nexport interface FilledFormData {\r\n [fieldName: string]: any;\r\n}\r\n\r\nexport interface VoiceFormRecorderResult {\r\n success: boolean;\r\n data?: FilledFormData;\r\n error?: string;\r\n transcription?: string;\r\n jsonForm?: string;\r\n}\r\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"service-providers.types.js","sourceRoot":"","sources":["../../src/types/service-providers.types.ts"],"names":[],"mappings":"","sourcesContent":["export interface SpeechToTextProvider {\r\n transcribe(audioBlob: Blob, lang: string): Promise<string>;\r\n}\r\n\r\nexport interface LLMProvider {\r\n fillFormFromTranscription(transcription: string, schema: any): Promise<any>;\r\n fillFormFromJson(json: string, schema: any): Promise<any>;\r\n}\r\n\r\nexport interface ServiceProviderConfig {\r\n speechToText?: {\r\n provider: 'whisper' | 'custom';\r\n apiKey?: string;\r\n baseUrl?: string;\r\n };\r\n llm?: {\r\n provider: 'openai' | 'custom';\r\n apiKey?: string;\r\n model?: string;\r\n baseUrl?: string;\r\n };\r\n}\r\n"]}
|